Java Review (十六、面向对象----Lambda 表达式)
@
Lambda 允许把函数作为一个方法的参数(函数作为参数传递到方法中)。
Lambda表达式入门
用匿名内部类实现命令模式
public interface Command {
//接口里定义的方法相当于处理封装行为
int progress(int[] target);
}
public class ProgressArray {
public ProgressArray() {
}
public void progress(int[] target,Command cmd) {
cmd.progress(target);
}
}
public class CommandInnerTest {
public CommandInnerTest() {
}
public static void main(String[] args) {
ProgressArray progressArray=new ProgressArray();
int[] array= new int[] {3, -4, 6, 4} ;
//处理数组,具体处理行为取决于匿名内部类
progressArray.progress(array, new Command() {
@Override
public void progress(int[] target) {
int sum=0;
int[] a=target;
for(int j=0;j<a.length;j++ ) {
int temp=a[j];
sum+=temp;
}
System.out.println("数组元素的总和:"+sum);
}
});
}
}
Lambda表示式实现命令模式
Lambda 表达式可用于简化创建匿名内部类对象:
public class CommandLambdaTest {
public static void main(String[] args) {
ProgressArray progressArray = new ProgressArray();
int[] array = { 3, -4, 6, 4 };
// 处理数组,具体行为取决于匿名内部类
progressArray.progress(array, (int[] target) -> {
int sum = 0;
for (int temp : array) {
sum += temp;
}
System.out.println("数组元素总和为" + sum);
});
}
}
当使用 Lambda 表达式代替匿名内部类创建对象时, Lambda 表达式的代码
块将会代替实现抽象方法的方法体, Lambda 表达式就相当一个匿名方法。
从上面语法格式可以看出, Lambda 表达式的主要作用就是代替匿名内部类的烦琐语法。它分三部分组成。
- 形参列表 。形参列表允许省略形参类型 。如果形参列表中只有一个参数,甚至连形参列表的圆括号也可以省略。
- 箭头 —>。必须通过英文中画线和大于符号组成。
- 代码块。如果代码块只包含一条语句, Lambda 表达式允许省略代码块的花括号,那么这条语句就不要用花括号表示语句结束。 Lambda 码块只有一条 return语句,甚至可以省略return关键字, Lambda 表达式会自动返回这条语句的值。
public class LambdaTest {
public static void main(String[] args) {
LambdaTest tester=new LambdaTest();
// 类型声明
MathOperation addition = (int a, int b) -> a + b;
// 不用类型声明
MathOperation subtraction = (a, b) -> a - b;
// 大括号中的返回语句
MathOperation multiplication = (int a, int b) -> {
return a * b;
};
// 没有大括号及返回语句
MathOperation division = (int a, int b) -> a / b;
System.out.println("10 + 5 = " + tester.operate(10, 5, addition));
System.out.println("10 - 5 = " + tester.operate(10, 5, subtraction));
System.out.println("10 x 5 = " + tester.operate(10, 5, multiplication));
System.out.println("10 / 5 = " + tester.operate(10, 5, division));
// 不用括号
GreetingService greetService1 = message -> System.out.println("Hello " + message);
// 用括号
GreetingService greetService2 = (message) -> System.out.println("Hello " + message);
greetService1.sayMessage("Runoob");
greetService2.sayMessage("Google");
}
interface MathOperation {
int operation(int a, int b);
}
interface GreetingService {
void sayMessage(String message);
}
private int operate(int a, int b, MathOperation mathOperation) {
return mathOperation.operation(a, b);
}
}
Lambda 表达式与函数式接口
Lambda 表达式的类型,也被称为"目标类型( target type) ", Lambda 表达式的目标类型必须是"函数式接口( functional interface ) "。 函数式接口代表只包含一个抽象方法的接口 。 函数式接口可以包含多个默认方法、类方法,但只能声明一个抽象方法 。
如果采用匿名内部类语法来创建函数式接口的实例,则只需要实现一个抽象方法,在这种情况下即可采用 Lambda 表达式来创建对象,该表达式创建出来的对象的目标类型就是这个函数式接口 。 查询 Java 8的 API 文档,可以发现大量的函数式接口,例如: Runnable、 ActionListener 等接口都是函数式接口 。
Java 8专门为函数式接口提供了 @FunctionalInterface 注解,该注解通常放在接口定义前面,该注解对程序功能没有任何作用,它用于告诉编译器执行更严格检查一一检查该接口必须是函数式接口 ,否则编译器就会报错。
由于 Lambda 表达式的结果就是被当成对象 , 因此程序中完全可以使用 Lambda 表达式进行赋值,
例如如下代码 :
// Runnable 接口中只包含一个无参数的方法
// Lambda 表达式代表的匿名方法实现了 Runnable 接口中唯一的、无参数的方法
// 因此下面的 Lambda 表达式创建了 一个 Runnable 对象
Runnable r = () -> {
for(int i = 0 ; i < 100 ; i ++){
System.out.println() ;
}
}
Lambda 表达式实现的是匿名方法,因此它只能实现特定函数式接口中的唯一方法 。 这意味着 Lambda 表达式有如下两个限制 :
- Lambda 表达式的目标类型必须是明确的函数式接口 。
- Lambda 表达式只能为函数式接口创建对象 。 Lambda 表达式只能实现一个方法 , 因此它只能为只有一个抽象方法的接口(函数式接口)创建对象 。
为了保证 Lambda 表达式的目标类型是一个明确的函数式接口,可以有如下三种常见方式 。
- 将 Lambda 表达式赋值给函数式接口类型的变量 。
- 将 Lambda 表达式作为函数式接口类型的参数传给某个方法 。
- 使用函数式接口对 Lambda 表达式进行强制类型转换。
Object obj1 = (Runnable) () - > {
for(int i = 0 ; i < 100 ; i ++{
System.out.println(i);
}
方法引用与构造器引用
如果 Lambda 表达式的代码块只有一条代码,程序就可以省略Lambda 表达式中
代码块的花括号 。 不仅如此,如果 Lambda 表达式的代码块只有一条代码,还可以在代码块中使用方法引用和构造器引用 。
方法引用和构造器引用可以让 Lambda 表达式的代码块更加简洁 。 方法引用和构造器引用都需要使用两个英文冒号。
引用类方法
//函数式接口
@FunctionalInterface
interface Converter{
//将 String 参数转换为Integer
Integer convert(String from);
}
// 下面代码使用 Larnbda 表达式创建 Converter 对象
Converter converterl = from -> Integer.valueOf(from);
//调用 converterl 对象的 convertO方法将字符串转换为整数
Integer val = converterl.convert( " 99 " );
System.out.println (val); // 输出整数 99
可以如下替换:
//方法引用代替 Larnbda 表达式:引用类方法
//函数式接口中被实现方法的全部参数传给该类方法作为参数
Converter converterl = Integer::valueOf;
引用特定对象的实例方法
// 下面代码使用 Larnbda 表达式创建 Converter 对象
Converter converter2 = from - > " fkit.org ".indexOf (from) ;
//调用 converterl 对象的 convertO方法将字符串转换为整数
Integer value = converter2.convert( "it "};
System.out.println(value} ; //输出 2
可以如下替换:
// 方法引用代替 Lambda 表达式 : 引用特定对象 的实例方法
// 函数式接口中被实现方法的全部参数传给该方法作为参数
Converter converter2 = "fkit.org": :indexOf ;
引用某类对象的实例方法
//定义函数式接口
@Functionallnterface
interface MyTest{
//根据 String 、 int 、 int 三个参数生成一个 String返回值
String test(String a , int b , int c);
}
// 下面代码使用 Lambda 表达式创建 MyTest 对象
MyTest mt = (a , b , c) - > a.substring(b , c};
//调用mt的test()方法
String str = mt.test( "Java 1 Love you" , 2 , 9) ;
System.out.println(str} ; // 输出 : va 1 Lo
可以如下替换:
// 方法引用代替 Lambda 表达式: 引 用某类对象 的实例方法
// 函数式接口中被实现方法的第一个参数作为调用者
// 后面的参数全部传给该方法作为参数
MyTest mt = String ::substring;
引用构造器
@Funct ionallnterface
interface YourTest{
//根据 String 参数生成一个 JFrame 返回值
JFrame win(String title);
// 下面代码使用 Lambda 表达式创建 YourTest 对象
YourTest yt = (String a) - > new JFrame(a);
JFrame jf = yt.win( "我的窗口 ") ;
System.out.println(jf) ;
可以如下替换:
// 构造器引用代替 Lambda 表达式
//函数式接口中被实现方法的全部参数传给该构造器作为参数
YourTest yt = JFrame ::new;
Lambda 表达式与匿名内部类的联系和区别
从前面介绍可以看出, Lambda 表达式是匿名内部类的一种简化 , 因此它可以部分取代匿名内部类的作用。
Lambda 表达式与匿名内部类存在如下相同点 :
- Lambda 表达式与匿名内部类一样,都可以直接访问 "effectively final" 的局部变量,以及外部类的成员变量(包括实例变量和类变量〉 。
- Lambda 表达式创建的对象与匿名内部类生成的对象一样,都可以直接调用从接口中继承的默认方法 。
Lambda 表达式与匿名内部类主要存在如下区别:
- 匿名内部类可以为任意接口创建实例一一不管接口包含多少个抽象方法,只要匿名内部类实现所有的抽象方法即可;但 Lambda 表达式只能为函数式接口创建实例。
- 匿名内部类可以为抽象类甚至普通类创建实例 ; 但 Lambda 表达式只能为函数式接口创建实例 。
- 匿名内部类实现的抽象方法的方法体允许调用接口中定义的默认方法 ; 但 Lambda 表达式的代码块不允许调用接口中定义的默认方法。
参考:
【1】:《疯狂Java讲义》
【2】:https://www.jb51.net/article/171183.htm
【3】:https://www.runoob.com/java/java8-lambda-expressions.html
【4】:《Java核心技术 卷一》