java8 lambda 表达式详解

时间:2022-09-07 12:14:37
  • 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 追踪代码运行过程中的堆内存,发现会生成以下实例:

    java8 lambda 表达式详解

    生成的实例类名为 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 可转换为 Exception
    throw 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

 

java8 lambda 表达式详解的更多相关文章

  1. Java8 Lambda表达式详解手册及实例

    先贩卖一下焦虑,Java8发于2014年3月18日,距离现在已经快6年了,如果你对Java8的新特性还没有应用,甚至还一无所知,那你真得关注公众号"程序新视界",好好系列的学习一下 ...

  2. JAVA8之lambda表达式详解

    原文:http://blog.csdn.net/jinzhencs/article/details/50748202 lambda表达式详解 一.问题 1.什么是lambda表达式? 2.lambda ...

  3. Lambda表达式详解(例子详解)(转自:http&colon;&sol;&sol;blog&period;csdn&period;net&sol;damon316&sol;article&sol;details&sol;51734661)

    Lambda表达式详解(例子详解)     lambda简介 lambda运算符:所有的lambda表达式都是用新的lambda运算符 " => ",可以叫他,“转到”或者 ...

  4. Java 8 Lambda 表达式详解

    一.Java 8 Lambda 表达式了解 参考:Java 8 Lambda 表达式 | 菜鸟教程 1.1 介绍: Lambda 表达式,也可称为闭包,是推动 Java 8 发布的最重要新特性. La ...

  5. 类型:&period;net;问题:C&num;lambda表达式;结果:Lambda表达式详解

    Lambda表达式详解   前言 1.天真热,程序员活着不易,星期天,也要顶着火辣辣的太阳,总结这些东西. 2.夸夸lambda吧:简化了匿名委托的使用,让你让代码更加简洁,优雅.据说它是微软自c#1 ...

  6. Java中lambda表达式详解

    原文地址:http://blog.laofu.online/2018/04/20/java-lambda/ 为什么使用lambda 在java中我们很容易将一个变量赋值,比如int a =0;int ...

  7. Lambda表达式详解

    前言 1.天真热,程序员活着不易,星期天,也要顶着火辣辣的太阳,总结这些东西. 2.夸夸lambda吧:简化了匿名委托的使用,让你让代码更加简洁,优雅.据说它是微软自c#1.0后新增的最重要的功能之一 ...

  8. C&num; Lambda表达式详解,及Lambda表达式树的创建

    最近由于项目需要,刚刚学完了Action委托和Func<T>委托,发现学完了委托就必须学习lambda表达式,委托和Lambda表达式联合起来,才能充分的体现委托的便利.才能使代码更加简介 ...

  9. Lambda表达式详解(转载)

    原文链接:http://www.cnblogs.com/knowledgesea/p/3163725.html lambda简介 lambda运算符:所有的lambda表达式都是用新的lambda运算 ...

随机推荐

  1. 谨慎使用Sql server data tool 架构比对排除

      现象:某个架构始终不能更新     VS中使用SSDT很方便,进行架构比对时,可以选择性的更新.但在排除操作上,要相当谨慎.往往排除的并不是一个差异项. 如下图,表或视图的差异,如果有多处,可以选 ...

  2. Visual Studio CLR Profiler

    http://blogs.msdn.com/b/dotnet/archive/2013/04/04/net-memory-allocation-profiling-with-visual-studio ...

  3. 数据库左连接left join、右连接right join、内连接inner join on 及 where条件查询的区别

    join on 与 where 条件的执行先后顺序: join on 条件先执行,where条件后执行:join on的条件在连接表时过滤,而where则是在生成中间表后对临时表过滤 left joi ...

  4. jQuery学习小结3——AJAX

    一.jQuery的Ajax方法 jQuery对Ajax 做了大量的封装,使用起来也较为方便,不需要去考虑浏览器兼容性.对于封装的方式,jQuery 采用了三层封装: 最底层的封装方法为——$.ajax ...

  5. Flex-box 学习

    .flex-cont{ /*定义为flexbox的“父元素”*/ display: -webkit-box; display: -webkit-flex; display: flex; /*子元素沿主 ...

  6. Seven Python Tools All Data Scientists Should Know How to Use

    Seven Python Tools All Data Scientists Should Know How to Use If you’re an aspiring data scientist, ...

  7. Gtk中的文本视图(GtkTexViewWidget)

    Gtk中的文本视图(GtkTexViewWidget) Gtk中的文本视图(GtkTexView Widget) 在本章的Gtk+程序设计教程中,我们将重点介绍 GtkTexView 构件. GtkT ...

  8. SQL Server中CURD语句的锁流程分析

    我只在数据库选项已开启“行版本控制的已提交读”(READ_COMMITTED_SNAPSHOT为ON)中进行了观察. 因此只适用于这种环境的数据库. 该类数据库支持四种不同事务隔离级别,下面分别观察数 ...

  9. 用python快速搭建WEB服务器

    cmd下进入你要搞WEB项目的目录 输入↓方代码 python -m SimpleHTTPServer 端口号# 默认是8000 这样就启动了一个简单的WEB服务器

  10. BOOTICE&lpar;引导扇区维护工具&rpar; V1&period;3&period;3 中文免费绿色版

    软件名称: BOOTICE(引导扇区维护工具)软件语言: 简体中文授权方式: 免费软件运行环境: Win7 / Vista / Win2003 / WinXP 软件大小: 357KB图片预览: 软件简 ...