当前位置: 首页 > 编程日记 > 正文

java8 lambda 表达式详解

  • lambada 表达式实质上是一个匿名方法,但该方法并非独立执行,而是用于实现由函数式接口定义的唯一抽象方法
  • 使用 lambda 表达式时,会创建实现了函数式接口的一个匿名类实例
  • 可以将 lambda 表达式视为一个对象,可以将其作为参数传递

1. 函数式接口

函数式接口是仅含一个抽象方法的接口,但可以指定 Object 定义的任何公有方法。

  • 以下是一个函数式接口:
1 @FunctionalInterface
2 public interface IFuntionSum<T extends Number> {
3     T sum(List<T> numbers);      // 抽象方法
4 }

  • 以下也是一个函数式接口:
1 @FunctionalInterface
2 public interface IFunctionMulti<T extends Number> {
3     void multi(List<T> numbers); // 抽象方法
4     
5     boolean equals(Object obj);  // Object中的方法
6 }

  • 但如果改为以下形式,则不是函数式接口:
1 @FunctionalInterface
2 public interface IFunctionMulti<T extends Number> extends IFuntionSum<T> {
3     void multi(List<T> numbers);
4     
5     @Override
6     boolean equals(Object obj);
7 }
8 // IFunctionMulti 接口继承了 IFuntionSum 接口,此时 IFunctionMulti 包含了2个抽象方法

tip 1: 可以用 @FunctionalInterface 标识函数式接口,非强制要求,但有助于编译器及时检查接口是否满足函数式接口定义
tip 2: 在 Java 8 之前,接口的所有方法都是抽象方法,在 Java 8 中新增了接口的默认方法

2. lambda 表达式

  • lambda 表达式的2种形式
    包含单独表达式 :parameters -> an expression

     1 list.forEach(item -> System.out.println(item)); 
    

    包含代码块:parameters -> { expressions };

    list.forEach(item -> {int numA = item.getNumA();int numB = item.getNumB();System.out.println(numA + numB);
    });

    左侧指定 lambda 表达式需要的参数,右侧指定 lambda 方法体

  • 上文提到,lambda 无法独立执行,它必须是实现一个函数式接口的唯一抽象方法。
    每个 lambda 表达式背后必定有一个函数式接口,该表达式实现的是这个函数式接口内部的唯一抽象方法。

    譬如以下 lambda 表达式:

    list.forEach(item -> System.out.println(item));

    我们看 ArrayList 中 foreach 方法:

    @Override
    public void forEach(Consumer<? super E> action) {// 太长了,不看了~
    }

    其中 Consumer 是一个函数式接口:

    @FunctionalInterface
    public interface Consumer<T> {void accept(T t);    // lambda 表达式 item -> System.out.println(item) 实现了该方法default Consumer<T> andThen(Consumer<? super T> after) {Objects.requireNonNull(after);return (T t) -> { accept(t); after.accept(t); };}
    }

    Consumer 接口是 Java 8 中预先定义的函数式接口,java.util.function 包下都是些预定义的函数式接口
    function 包下的部分接口使用了泛型,具有很强的通用性,在自定义函数式接口前,不妨去这个包下找找有没有能用的

  • 在执行 lambda 表达式时,会自动创建一个实现了目标函数式接口的类实例,该类实例是一个匿名内部类。

    同样以 list 的 foreach 方法为例:

    @FunctionalInterface
    public interface Consumer<T> {void accept(T t);    // lambda 表达式 item -> System.out.println(item) 实现了该方法default Consumer<T> andThen(Consumer<? super T> after) {Objects.requireNonNull(after);return (T t) -> { accept(t); after.accept(t); };}
    }

    用 Java VisualVM 追踪代码运行过程中的堆内存,发现会生成以下实例:

    生成的实例类名为 LambdaDemo$$Lambda$1,根据匿名内部类的命名规则可知,这是 LamddaDemo的 一个匿名内部类。(普通匿名内部类 和 lambda匿名内部类的命名规则见下文)。
  • 同样,由于 lambda 表达式在执行时会生成目标函数式接口的类实例,因此我们可以做以下操作:

    // 有以下函数式接口
    @FunctionalInterface
    public interface IFuntionSum<T extends Number> {T sum(List<T> numbers);
    }// 将一个lambda表达式赋值给函数式接口引用(类型须兼容)
    IFuntionSum<Long> function = list -> {Long sum = 0L;for (Long item : list) {sum += item;}return sum;
    };
    function.sum(Arrays.asList(1L, 2L));   // 执行结果为3L

    在开发过程中,我们可以将 lambda 表达式等同于一个对象使用,对其声明、引用、传递。

  • 匿名内部类 和 lambda 表达式匿名内部类的命名规则

    内部类的命名规则:外部类名 + $ + 内部类名
    匿名类的命名规则:外部类名 + $ + (1, 2, 3,第几个匿名类就显示几)
    lambada 匿名内部类的命名规则:外部类名 + $$ + Lambda + $ + (1, 2, 3,第几个lambda表达式就显示几)

    假设外部类中用到了2个lambda 表达式,则生成的2个匿名内部类的命名分别为 :

    外部类名$$Lambda$1 和 外部类名$$Lambda$2
    

3. lambda 表达式规约

  • lambda 表达式的参数可以通过上下文推断,如果需要显示声明一个参数的类型,则必须为所有的参数声明类型。

    @FunctionalInterface
    public interface IFunctionMod {boolean (int n, int d);
    }IFunctionMod function = (n, d) -> (n % d) == 0          // 合理,n 和 d 的类型通过上下文推断
    IFunctionMod function = (int n, int d) -> (n % d) == 0   // 合理,指定 n 和 d 的类型
    IFunctionMod function = (int n, d) -> (n % d) == 0       // 不合理,须显示声明所有参数类型

  • lambda 表达式中抛出的异常需要与目标函数式接口的抽象方法抛出的异常类型兼容:
    以下是合理的:

    @FunctionalInterface
    public interface IFunctionMod {boolean (int n, int d) throw Exception; 
    }IFunctionMod function = (n, d) -> {if (d == 0) {// IOException是EXception 的子类,通过类型转换,IOException 可转换为 Exceptionthrow new IOException("test"); } return n % d == 0;
    };

    如果反一下,就不行了:

    @FunctionalInterface
    public interface IFunctionMod {boolean (int n, int d) throw IOException; 
    }IFunctionMod function = (n, d) -> {if (d == 0) {// 父类不能通过自动类型转换转为子类,lambda 表达式抛出的异常类型与抽象方法抛出的异常类型不兼容throw new Exception("test");} return n % d == 0;
    };

  • lambda 表达式中参数类型需要与目标函数式接口中抽象方法的参数类型兼容。

    tip :从接口与实现的角度,可以很容易理解抛出异常兼容 和 参数类型兼容 这2点。

4. 方法引用

可以引用已有方法构造 lambda 表达式,这里给一个例子,不做详细解释:

list.forEach(System.out::print)

原文作者:EricAlpha:https://www.jianshu.com/p/613a6118e2e0

转载于:https://www.cnblogs.com/gu-bin/p/11225749.html

相关文章:

【基础知识】win10常用快捷键

1、winctrlD 创建虚拟桌面 2、winctrl左右箭头 切换桌面 winctrlF4 关闭当前虚拟桌面 (按wintab 可以通过鼠标操作虚拟桌面的创建、切换与删除) 3、winM 全部窗口最小化 (winD 可在最小化全部窗口与恢复全部窗口状态 间自由切换 4、win数字 打开任务栏中从左到右的应用&#xf…

ASP.NET2.0-防止同一用户同时登陆

要防止同一用户同时登陆,首页应该记录在线用户的信息(这里与用户名为例),然后判断正在登陆的用户里面是否已存在。在这里使用一个cache存放已经登陆的用户名&#xff0c;但是还有一个问题就是要知道用户是什么时候离开系统的呢&#xff1f;这就要定期清除cache中的内容了&#…

如何选择真正的万兆防火墙?

http://www.cnw.com.cn/security-10g-firewall/htm2011/20110526_225444.shtml 随着“三网合一”、 “P2P视频”、“高清宽带”、“云时代”等逐渐成为人们关注的焦点&#xff0c;网络带宽的需求产生了几何级别的增长。目前&#xff0c;“千兆到桌面、万兆做骨干”对于交换机和…

Salesforce Lightning开发学习(二)Component组件开发实践

lightning的组件区分标准组件、自定义组件和AppExchange组件。标准组件由SF提供&#xff0c;自定义组件由developer自行开发&#xff0c;AppExchange组件由合作伙伴建立。下面我们写一个简单的列表demo来学习下自定义开发组件Component。 第一步&#xff1a;新建一个对象&#…

转 java c++互传arraylist

Java JNI由浅入深(包括&#xff1a;Java和C互传ArrayList泛型对象参数) 2010-11-25 09:57 1694人阅读 评论(1) 收藏 举报我们知道Java是一个运行在虚拟机里面的高级的编程语言&#xff0c;如果要调用系统的动态链接库的话&#xff0c;就要先声明native修饰的方法(类似接口里面的…

GoAccess安装及分析nginx实时日志

GoAccess是一个基于终端的快速日志分析器。其核心思想是实时快速分析和查看Web服务器统计信息&#xff0c;而无需使用您的浏览器&#xff08;如果您希望通过SSH快速分析访问日志&#xff0c;或者只是喜欢在终端中工作&#xff09;&#xff0c;终端输出是默认输出&#xff0c;但…

在写游戏时钟类时,应确保时钟的计算是以某个固定的CPU为标准的

在写游戏时钟类时&#xff0c;应确保时钟的计算是以某个固定的CPU为标准的。下面通过一个例子来阐述如何实现这一思想&#xff1a; // 记录特定进程的affinity maskunsigned long lProcessAffinityMask;// 记录系统的affinity maskunsigned long lSystemAffinityMask;::GetProc…

学习Spring Boot

Spring boot 是什么 ? 简单说, spring boot 是一个构建项目的工具, 一个脚手架. Spring boot 能干什么? spring boot 做非常少的配置就可以构建生产级别的单体应用. Spring boot 怎么干的? 下面让我们来用spring boot 做一个hello world. 环境准备, 需要保证你的机器上已经有…

Linux添加用户组和添加用户

1.用户组 添加组&#xff1a;groupadd 组名 [rootServer-n93yom ~]# groupadd dev [rootServer-n93yom ~]# cat /etc/group | grep dev dev:x:10011: [rootServer-n93yom ~]# 删除组&#xff1a;groupdel 组名 [rootServer-n93yom ~]# groupdel dev [rootServer-n93yom ~]# cat…

用javascript实现仿163的js广告向下挤压页面的效果

本次实现一个js小小特效&#xff0c;效果就是广告从页面的上方出来将基本页面挤压下去。 实现的思路是将两个div&#xff08;广告div1和div2&#xff09;。将div1的高度不断增加&#xff0c;增加的同时div2的y轴坐标变大&#xff0c;div1的高度增加多少相对应的y轴坐标加大多少…

【eclipse】快速调整eclipse背景和格式的方法

第一步 第二步 第三步 &#xff1a;选择相应的格式 效果如图

对做C#自定义控件的一点心得

近期在做DSOFramer这个控件&#xff0c;打算自己弄一个自定义控件来封装这个COM组件&#xff0c;中间遇到很多曲折&#xff0c;研究了一个星期&#xff0c;终于完成了 下面总结一下我做DSOFramer这个自定义控件的注意地方&#xff1a; 1、在创建一个Windows窗体控件库的时候&am…

numpy知识点

1、nonzero 对于一维数据来说&#xff0c;将返回符合条件的 下标 >>> b1 np.array([True, False, True, False]) >>> np.nonzero(b1)(array([0, 2]),)对于二维数据来说&#xff0c;将返回两维 元组&#xff0c; 第一维是符合条件的 x的索引&#xff0c;第二…

Excel导入异常Cannot get a text value from a numeric cell解决

POI操作Excel时偶尔会出现Cannot get a text value from a numeric cell的异常错误。 异常原因&#xff1a;Excel数据Cell有不同的类型&#xff0c;当我们试图从一个数字类型的Cell读取出一个字符串并写入数据库时&#xff0c;就会出现Cannot get a text value from a numeric …

【蓝桥java】递归基础之反向输出字符串

题目&#xff1a;输入一个字符串&#xff0c;要求将该字符串反向输出 代码实现&#xff1a; package xn.zzunit.recurrence;/*** 递归方法反向输出字符串* author tyrantForever**/ public class Project1 {public static void main(String[] args) {String testString "…

SQL Server中灾难时备份结尾日志(Tail of log)的两种方法

简介 在数据库数据文件因各种原因发生损坏时&#xff0c;如果日志文件没有损坏。可以通过备份结尾日志&#xff08;Tail of log)使得数据库可以恢复到灾难发生时的状态。 例如: 上图中。在DB_1中做了完整备份&#xff0c;在Log_1,Log_2处做了日志备份。在Log_2备份之后不久&…

C examples

最近在看David R. Hanson 的《C Interfaces and Implementations》&#xff0c;文中第一章提到了Literate Programming 作者举了一个例子&#xff1a; 功能&#xff1a;用于检测输入中相邻且相同的单词 #include<stdio.h> #include<math.h> #include<errno.h&g…

特征工程:特征生成,特征选择(三)

转自&#xff1a;https://blog.csdn.net/cymy001/article/details/79169862 特征生成 特征工程中引入的新特征&#xff0c;需要验证它确实能提高预测得准确度&#xff0c;而不是加入一个无用的特征增加算法运算的复杂度。 1. 时间戳处理 时间戳属性通常需要分离成多个维度比如年…

JAVA设计模式之不变模式

在阎宏博士的《JAVA与模式》一书中开头是这样描述不变&#xff08;Immutable&#xff09;模式的&#xff1a; 一个对象的状态在对象被创建之后就不再变化&#xff0c;这就是所谓的不变模式。 不变模式的结构 不变模式可增强对象的强壮型(robustness)。不变模式允许多个对象共享…

【蓝桥java】递归基础之智力训练

题目&#xff1a; 匪警请拨110,即使手机欠费也可拨通&#xff01; 为了保障社会秩序&#xff0c;保护人民群众生命财产安全&#xff0c;警察叔叔需要与罪犯斗智斗勇&#xff0c;因而需要经常性地进行体力训练和智力训练&#xff01; 某批警察叔叔正在进行智力训练&#xff1a; …

EF-Entity Framework 相关技术点收集贴

不定期、偶尔、添加更新 在网络上看到或者自己开发过程中碰到的EF&#xff0d;Entity Framework相关技术点 本文地址:http://www.cnblogs.com/vnii/archive/2012/02/28/2371736.html 1.数据表字段有默认值&#xff0c;比如DateTime类型的字段CreateTime默认值为数据新增的时间g…

[翻译]LightRacer游戏架构

1.0版本的Light Racer架构可说的不多。仅有一个单一的Activity&#xff0c;进行按钮的处理&#xff0c;显示游戏相关数据和显示GameView。我将在另一篇文章中说明游戏的画面是如何工作的&#xff0c;但是现在我先声明一下的就是&#xff1a;在Android中&#xff0c;单个Activit…

case when里的like功能 ////// 截取(substr)

case when里的like功能 假如要用到case when又要用到like这样的功能&#xff0c;即如果字符串包含‘语文’就怎么怎么样&#xff0c;包含‘数学’就怎么怎么样&#xff0c;包含‘英语’就怎么怎么样&#xff0c;like是用于where中的&#xff0c;放在case when里面是无效的&…

在Asp.Net MVC中设定site路径所对应的默认action

设置路由的default的Controller和Action可以达到我们预期的效果&#xff0c;代码如下所示&#xff1a; public class RouteConfig {public static void RegisterRoutes(RouteCollection routes){routes.IgnoreRoute("{resource}.axd/{*pathInfo}");routes.MapRoute(n…

【蓝桥java】递归基础之输出连续数字

题目&#xff1a;使用递归方法输出连续的数字 代码实现&#xff1a; package xn.zzunit.recurrence;/*** 递归方法输出连续的数字* author tyrantForever**/ public class Project2 {public static void main(String[] args) {printNumber(2, 9);}public static void printNu…

慢慢学Linux驱动开发,第十章,GNU C的扩展

内核开发者使用的C语言涵盖了ISO C99标准和GNU C扩展特性。这里简单介绍一下GNU C的扩展特性。 1.内联&#xff08;inline&#xff09;函数 GNU的C编译器支持内联函数&#xff0c;也是C的一个特性之一。就是函数会在所调用的位置上展开&#xff0c;这样做虽然会导致代码量的增加…

vMA学习笔记之一:将vMA加入域

目的&#xff1a; 将vMA加入域的方法 操作步骤&#xff1a; 1、 开启vMA 2、 按住AltF2切换到虚拟终端界面&#xff0c;使用vi-admin用户登录 2 3、 已经登录进来了 4、 在进行加域操作之前&#xff0c;必须确保DNS配置正确&#xff08;在初始化安装的时候会提示你设置DNS&…

关于DWG文件转换成PDF

最近有这样一个需求&#xff0c;客户会提供DWG文件&#xff0c;因为DWG文件是不能直接在网页上显示的&#xff0c;所以必须对他做处理&#xff0c;要求是转换成PDF格式。我查了很久的资料&#xff0c;很多都是基于C#和.NET的方法&#xff0c;而且都是说的很模糊&#xff0c;不是…

剑指offer--day07

1.1 题目&#xff1a;反转链表&#xff1a;输入一个链表&#xff0c;反转链表后&#xff0c;输出新链表的表头。 1.2 思路&#xff1a;这道题&#xff0c;我们要做到的是反转链表&#xff0c;我们的思路是将前一个节点与后一个节点断开&#xff0c;然后让后一个节点指向前一个节…