JAVA Lambda 表达式
1. Lambda 表达式
1.1 函数是编程思想概述
在数学中,函数就是有输入量、输出量的一套计算方案,也就是“拿数据做操作”
面向对象思想强调 “ 必须通过对象的形式来做事情 ”
函数式思想则尽量忽略面向对象的复杂语法:“ 强调做什么,而不是什么形式去做 ”
Lambda 表达式就是函数式思想的体现
1.2 体验Lambda表达式
需求:启动一个线程,在控制台输出一句话:多线程程序启动了
方式1:
- 定义一个类
MyRunnable
实现Runnable
接口,重写run ()
方法 - 创建
MyRunnable
类的对象 - 创建
Thread
类的对象,把MyRunnable
的对象作为构造参数传递 - 启动线程
方式2:
- 匿名内部类的方式改进
方式3:
- Lambda 表达式的方式改进
//Lambda表达式的方法改进
new Thread( () -> {
System.out.println("多线程程序启动了");
} ).start();
1.3 Lambda 表达式的标准格式
匿名内部类中重写
run()
方法的代码分析
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("多线程程序启动了");
}
}).start();
- 方法形式参数为空,说明调用方法时不需要传递参数
- 返回值类型为 void,说明方法执行没有结果返回
- 方法中的内容,是我们具体要做的事情
Lambda 表达式的代码分析
//Lambda表达式的方法改进
new Thread( () -> {
System.out.println("多线程程序启动了");
} ).start();
- ( ) :里面没有内容,可以看成时方法形式参数为空
- -> :用箭头指向后面要做的事情
- { } :包含一段代码,我们称之为代码块,可以看成是方法体中的的内容
组成 Lambda 表达式的三要数:形式参数,箭头,代码块
Lambda 表达式的格式
- 格式:( 形式参数 ) -> { 代码块 }
- 形式参数:如果有多个参数,参数之间用逗号隔开;如果没有参数,留空即可
- -> :用英文中画线和大于符号组成,固定写法。代表指向性动作
- 代码块:是我们具体要做的事情,也就是以前我们写的方法体内容
1.4 Lambda 表达式的练习
Lambda 表达式的使用前提
- 有一个接口
- 接口中有且只有一个抽象方法
练习1:
- 定义一个接口 ( Eatable ) ,里面定义一个吹嘘方法:
void eat();
- 定义一个测试类 ( EatableDemo ) ,在测试类中提供两个方法
- 一个方法是:useEatable ( Eatable e )
- 一个方法是主方法,在主方法中调用
useEatable
方法
/*
Lambda 表达式的格式:( 形式参数 ) -> { 代码块 }
练习1:
定义一个接口 ( Eatable ) ,里面定义一个吹嘘方法:void eat();
定义一个测试类 ( EatableDemo ) ,在测试类中提供两个方法
一个方法是:useEatable ( Eatable e )
一个方法是主方法,在主方法中调用 useEatable 方法
*/
public class EatableDemo {
public static void main(String[] args) {
//在主方法中调用 useEatable 方法
Eatable e = new EatableImpl();
useEatable(e);
//匿名内部类
useEatable(new Eatable() {
@Override
public void eat() {
System.out.println("一天一苹果,医生远离我");
}
});
//Lambda 表达式改进
useEatable(() -> {
System.out.println("一天一苹果,医生远离我");
});
}
private static void useEatable(Eatable e) {
e.eat();
}
}
public class EatableImpl implements Eatable {
@Override
public void eat() {
System.out.println("一天一苹果,医生远离我");
}
}
public interface Eatable {
void eat();
}
练习二
- 定义一个接口(Flyable),里面定义一个抽象方法:
void fly(String s);
- 定义一个测试类(FlyableDemo),在测试类中提供两个方法
- 一个方法是:
useFlyable(Flyable f)
- 一个方法是主方法:在主方法中调用
useFlyable
方法
- 一个方法是:
/*
练习二
1.定义一个接口(Flyable),里面定义一个抽象方法:void fly(String s);
2.定义一个测试类(FlyableDemo),在测试类中提供两个方法
一个方法是:useFlyable(Flyable f)
一个方法是主方法:在主方法中调用 useFlyable 方法
*/
public class FlyableDemo {
public static void main(String[] args) {
//在主方法中调用useFlyable方法
//匿名内部类
useFlyable(new Flyable() {
@Override
public void fly(String s) {
System.out.println(s);
System.out.println("海上冲浪");
}
});
System.out.println("------------------------");
//Lambda 表达式改进
useFlyable((String s) -> {
System.out.println(s);
System.out.println("网上冲浪");
});
}
private static void useFlyable(Flyable f) {
f.fly("乘风破浪");
}
}
public interface Flyable {
void fly(String s);
}
练习3
- 定义一个接口(Addabel),里面定义一个抽象方法:
int add(int x, int y);
- 定义一个测试类(AddableDemo),在测试类中提供两个方法
- 一个方法是:
useAddable(Addable a)
- 一个方法是主方法,在主方法中调用
useAddable
方法
- 一个方法是:
/*
练习3
1.定义一个接口(Addabel),里面定义一个抽象方法: int add(int x, int y);
2.定义一个测试类(AddableDemo),在测试类中提供两个方法
一个方法是:useAddable(Addable a)
一个方法是主方法,在主方法中调用 useAddable 方法
*/
public class AddableDemo {
public static void main(String[] args) {
//在主方法中调用useAddable方法
useAddable((int x, int y) -> {
return x + y;
// return x - y;
// return x * y;
// return x / y;
});
}
private static void useAddable(Addable a) {
int sum = a.add(10, 20);
System.out.println(sum);
}
}
public interface Addable {
int add(int x, int y);
}
1.5 Lambda 表达式的省略模式
省略规则:
- 参数类型可以省略。但是又多个参数的情况下,不能只省略一个
- 如果参数有且只有一个,那么小括号可以省略
- 如果代码块的语句只有一条,可以省略大括号和分号,甚至是return
//参数的类型可以省略
useAddable((x, y) -> {
return x + y;
});
//但是有多个参数的情况下,不能只省略一个
useAddable((x, int y) -> {
return x + y;
} );
useFlyable((s) -> {
System.out.println(s);
});
//如果有且仅有一个,那么小括号可以省略
useFlyable(s -> {
System.out.println(s);
});
//如果代码块的语句只有一条,可以省略大括号和分号
useFlyable(s -> System.out.println(s));
//如果代码块的语句只有一条,可以省略大括号和分号如果有return,return也要省略
useAddable((x, y) -> x + y);
1.6 Lambda 表达式的注意事项
注意事项
- 使用
Lambda
必须要有接口,并且要求接口中有且只有一个抽象方法 - 必须有上下文环境,才能推导出
Lambda
对应的接口- 根据 局部变量的赋值 得知 lambda 对应的接口:
Runnable r = () -> System.out.println("Lambda表达式");
- 根据 调用方法的参数 得知 Lambda 对应的接口:
new Thread(() -> System.out.println("Lambda表达式")).start();
- 根据 局部变量的赋值 得知 lambda 对应的接口:
1.7 Lambda 表达式和匿名内部类的区别
所需类型不同
- 匿名内部类:可以是接口,也可以是抽象类,还可以是具体类
- Lambda 表达式:只能是接口
使用限制不同
- 如果接口中有且仅有一个抽象方法,可以使用Lambda表达式,也可以使用匿名内部类
- 如果接口中多余一个抽象方法,只能使用匿名内部类,而不能使用Lambda表达式
实现原理不同
- 匿名内部类:编译之后,产生一个单独的
.class
字节码文件 - Lambda 表达式:编译之后,没有一个单独的
.class
字节码文件。对应的字节码会在运行的时候动态生成
a表达式")).start();`
1.7 Lambda 表达式和匿名内部类的区别
所需类型不同
- 匿名内部类:可以是接口,也可以是抽象类,还可以是具体类
- Lambda 表达式:只能是接口
使用限制不同
- 如果接口中有且仅有一个抽象方法,可以使用Lambda表达式,也可以使用匿名内部类
- 如果接口中多余一个抽象方法,只能使用匿名内部类,而不能使用Lambda表达式
实现原理不同
- 匿名内部类:编译之后,产生一个单独的
.class
字节码文件 - Lambda 表达式:编译之后,没有一个单独的
.class
字节码文件。对应的字节码会在运行的时候动态生成