都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表达式来写。
函数式接口举例
自定义函数式接口
作为参数传递 Lambda 表达式
Java 内置四大核心函数式接口
方法引用与构造器引用
方法引用(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表达式的描述及应用举例,行文匆匆,如有不足及错误之处,欢迎指正。