Lambda表达式是Java8的重要更新,Lambda表达式支持将代码块作为方法参数,Lambda表达式允许使用更简洁的代码来创建只有一个抽象方法的接口的实例,这种接口被称为函数式接口。
7.1 Lambda表达式
Lambda有三部分组成:
- 形参列表:形参列表允许省略形参类型。如果形参列表形参列表中只有一个参数,甚至连形参列表的圆括号也可以省略。
- 箭头:必须通过英文中画线和大于符号组成。
- 代码块:如果代码块只包含一条语句,Lambda表达式允许省略代码块的花括号。Lambda表达式只有一条return语句,可以省略return关键字。
7.2 Lambda表达式与函数式接口
Lambda表达式实际上会被当成一个“任意类型”的对象,到底需要当成何种类型的对象,取决于运行环境的需要。因为Lambda表达式的结果被当成任意类型的对象,所以程序中可以使用Lambda表达式赋值。但Lambda表达式的目标类型必须是“函数式接口”。
函数式接口代表只包含一个抽象方法的接口。函数式接口可以包含多个默认方法、类方法,但只能包含一个抽象方法。Java8为函数式接口提供了@FunctionInterface注解,提醒编译器进行更严格的检查。
Lambda表达式有如下限制:
- Lambda表达式的目标类型必须是明确的函数式接口。
- Lambda表达式只能为函数式接口创建对象。如Object o = () -> {}是不对的,因为Object不是函数式接口,但Object o = (Runnable) () -> {}可以。
保证Lambda表达式的目标类型是函数式接口的三种方式:
- 将Lambda表达式赋值给函数式接口类型的变量。
- 将Lambda表达式作为函数式接口类型的参数传给某个方法。
- 使用函数式接口对Lambda表达式进行强制类型转换。
java.util.Function包下的主要函数式接口:
- XxxFunction:包含一个apply()方法,用于对指定数据进行转换处理,并返回一个新的值。
- XxxConsumer:包含一个accept()方法,用于对数据进行处理,但不返回处理结果。
- XxxPredicate:包含一个test()方法,用于判断参数是否符合特定条件并进行筛选。
- XxxSupplier:包含一个getAsXxx()方法,不需要入参,根据某种逻辑返回一个数据。
7.3 方法引用与构造器引用
引用类方法:
- 格式:类名::类方法
- 说明:函数式接口中被实现方法的全部参数传给该类方法作为参数。
- 原Lambda表达式:(a, b, ...) -> 类名.类方法(a, b, ...)
- 举例:= param -> Interger.valueOf(param); 变为 = Integer::valueOf;
引用特定对象的实例方法:
- 格式:特定对象::实例方法
- 说明:函数式接口中被实现方法的全部参数传给该方法作为参数。
- 原Lambda表达式:(a, b, ...) -> 特定对象.实例方法(a, b, ...)
- 举例:= param ->"wang”.indexOf(param); 变为 = "wang"::indexOf;
引用某类对象的实例方法:
- 格式:类名::实例方法
- 说明:函数式接口中被实现方法的第一个参数作为调用者,后面的参数全部传给该方法作为参数。
- 原Lambda表达式:(a, b, ...) -> a.实例方法(b, ...)
- 举例:= (a, b, c) -> a.substring(b, c); 变为 = String::substring;
引用构造器:
- 格式:类名::new
- 说明:函数式接口中被实现方法的全部参数传给该构造器作为参数。
- 原Lambda表达式:(a, b, ...) -> new 类名(a, b, ...)
- 举例:= (String a) -> new JFrame(a); 变为 = JFrame::new;
7.4 Lambda表达式与匿名内部类的联系和区别
Lambda表达式与匿名内部类的相同点:
- 都可以直接访问“effectively final”的局部变量(即被匿名内部类访问时,默认变为final的局部变量)和外部类的成员变量。
- 生成的对象都可以直接调用从接口中继承的默认方法。
Lambda表达式与匿名内部类的区别:
- 匿名内部类可以为任意接口创建实例——不管接口包含多少个抽象方法,只要匿名内部类实现所有的抽象方法即可:但Lambda表达式只能为函数式接口创建实例。
- 匿名内部类可以为抽象类甚至普通类创建实例;但Lambda表达式只能为函数式接口创建实例。
- 匿名内部类实现的抽象方法的方法体允许调用接口中定义的默认方法;Lambda表达式的代码块不允许调用接口中定义的默认方法。因为Lambda可以转为任意类型的实例。