都2020年了,你还不会用Lambda表达式吗?

Java8新特性之lambda表达式

Java 8 (又称为 jdk 1.8) 是 Java 语言开发的一个主要版本。 Java 8 是oracle公司于2014年3月发布,可以看成是自Java 5 以 来最具革命性的版本。Java 8为Java语言、编译器、类库、开发 工具与JVM带来了大量新特性

为什么要使用lambda表达式

Lambda 是一个匿名函数,我们可以把 Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。使用它可以写出更简洁、更 灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升。

lambda表达式语法

Lambda 表达式:在Java 8 语言中引入的一种新的语法元素和操作符。这个操作符为 “->” , 该操作符被称为 Lambda 操作符 或箭头操作符。它将 Lambda 分为两个部分: 左侧:指定了 Lambda 表达式需要的参数列表 右侧:指定了 Lambda 体,是抽象方法的实现逻辑,也即 Lambda 表达式要执行的功能。

语法格式一:无参,无返回值

  • Runnable r1 = () -> {System.out.println("Hello,Lambda!");};

语法格式二:Lambda 需要一个参数,但是没有返回值。

  • Consumer con = (String str) -> {System.out.println(str);};

语法格式三:数据类型可以省略,因为可由编译器推断得出,称为“类型推断”

  • Consumer <string> con = (str) -> {System.out.println(str);};

语法格式四:Lambda 若只需要一个参数时,参数的小括号可以省略

  • Consumer <String> = str -> {System.out.println(str);};

语法格式五:Lambda 需要两个或以上的参数,多条执行语句,并且可以有返回

  • Comparator <Integer> com = (x,y) -> {System.out.println(实现函数式接口方法!); return Integer.compare(x,y);};

语法格式六:当 Lambda 体只有一条语句时,return 与大括号若有,都可以省略

  • Comparator <Integer > com = (x,y) -> Integer.compare(x,y);

类型推断

上述 Lambda 表达式中的参数类型都是由编译器推断得出的。Lambda 表达式中无需指定类型,程序依然可以编译,这是因为 javac 根据程序 的上下文,在后台推断出了参数的类型。Lambda 表达式的类型依赖于上下文环境,是由编译器推断出来的。这就是所谓的“类型推断”。

函数式(Functional)接口

什么是函数式接口

  • 只包含一个抽象方法的接口,称为函数式接口
  • 你可以通过 Lambda 表达式来创建该接口的对象。(若 Lambda 表达式 抛出一个受检异常(即:非运行时异常),那么该异常需要在目标接口的抽 象方法上进行声明)。
  • 我们可以在一个接口上使用 @FunctionalInterface 注解,这样做可以检 查它是否是一个函数式接口。同时 javadoc 也会包含一条声明,说明这个 接口是一个函数式接口。
  • 在java.util.function包下定义了Java 8 的丰富的函数式接口。

如何理解函数式接口

  • Java从诞生日起就是一直倡导“一切皆对象”,在Java里面面向对象(OOP) 编程是一切。但是随着python、scala等语言的兴起和新技术的挑战,Java不 得不做出调整以便支持更加广泛的技术要求,也即java不但可以支持OOP还 可以支持OOF(面向函数编程)

  • 在函数式编程语言当中,函数被当做一等公民对待。在将函数作为一等公民的 编程语言中,Lambda表达式的类型是函数。但是在Java8中,有所不同。在 Java8中,Lambda表达式是对象,而不是函数,它们必须依附于一类特别的 对象类型——函数式接口

  • 简单的说,在Java8中,Lambda表达式就是一个函数式接口的实例。这就是 Lambda表达式和函数式接口的关系。也就是说,只要一个对象是函数式接口的实例,那么该对象就可以用Lambda表达式来表示

  • 所以以前用匿名实现类表示的现在都可以用Lambda表达式来写

函数式接口举例

都2020年了,你还不会用Lambda表达式吗?

自定义函数式接口

都2020年了,你还不会用Lambda表达式吗?

作为参数传递 Lambda 表达式

都2020年了,你还不会用Lambda表达式吗?

Java 内置四大核心函数式接口

都2020年了,你还不会用Lambda表达式吗?

方法引用与构造器引用

方法引用(Method References)

  • 当要传递给Lambda体的操作,已经有实现的方法了,可以使用方法引用
  • 方法引用可以看做是Lambda表达式深层次的表达。换句话说,方法引用就是Lambda表达式也就是函数式接口的一个实例,通过方法的名字来指向一个方法,可以认为是Lambda表达式的一个语法糖
  • 要求:实现接口的抽象方法的参数列表和返回值类型,必须与方法引用的 方法的参数列表和返回值类型保持一致!
  • 格式:使用操作符 “::” 将类(或对象) 与 方法名分隔开来。
  • 如下三种主要使用情况: 1.对象::实例方法名 2.类::静态方法名 3.类::实例方法名

方法引用举例

例如: Consumer <Integer > con = x - > System.out.println(x);

等同于:Consumer <Integer > con1 = System.out :: println;

例如: Comparator < integer > com = (x,y) -> Integer.compare(x,y);

等同于: Comparator < integer > com1 = Integer :: compare;
    int value = com1.compare(10,11);

例如:BiPredicate <String > bp = (x,y) -> x.equals(y);

等同于:BiPredicate <String > bp1 = String :: equals;
    boolean flag = bp1.test("hello","hi");

  • 注意:当函数式接口方法的第一个参数是需要引用方法的调用者,并且第二 个参数是需要引用方法的参数(或无参数)时:ClassName::methodName

构造器引用

格式: ClassName::new

与函数式接口相结合,自动与函数式接口中方法兼容。 可以把构造器引用赋值给定义的方法,要求构造器参数列表要与接口中抽象方法的参数列表一致!且方法的返回值即为构造器对应类的对象。

例如:Function <Integer,MyClass > fun = n -> new MyClass(n);

等同于:Function <Integer,MyClass > fun1 = MyClass::new;

以上就是关于Java8中的一个新特性lambda表达式的描述及应用举例,行文匆匆,如有不足及错误之处,欢迎指正。