JavaSE Lambda 表达式
1. Lambda 表达式
1.1 函数式编程思想概述
在数学中,函数就是有输入量、输出量的一套计算方案,也就是 “拿数据做操作”
面向对象思想强调 “必须通过对象的形式来做事情”
函数式思想则尽量忽略面向对象的复杂语法,强调 “做什么,而不是以什么形式去做”
而我们要学的 Lambda 表达式就是函数式思想的体现
1.2 体验Lambda表达式
需求:启动一个线程,在控制台输出一句话:半仙儿努力考研
-
方式1:
- 定义一个类MyRunnable实现Runnable接口,重写run()方法
- 创建MyRunnable类的对象
- 创建Thread类的对象,把MyRunnable的对象作为构造参数传递
- 启动线程
-
方式2:
匿名内部类的方式改进 -
方式3:
Lambda表达式的方式改进
package FH.Lambda01;
public class MyRunnable implements Runnable{
@Override
public void run() {
System.out.println("半仙儿努力考研");
}
}
package FH.Lambda01;
/*
需求:
启动一个线程,在控制台输出一句话:半仙儿努力考研
*/
public class LambdaDemo {
public static void main(String[] args){
//实现类的方式实现需求
// MyRunnable my = new MyRunnable();
// Thread t = new Thread(my);
// t.start();
//匿名内部类的方式改进
// Thread t2 = new Thread(new Runnable() {
// @Override
// public void run() {
// System.out.println("半仙儿努力考研");
// }
// });
// t2.start();
//Lambda表达式的方式改进
// Thread t= new Thread( () -> {
// System.out.println("半仙儿努力考研");
// } );
// t.start();
new Thread( () -> {
System.out.println("半仙儿努力考研");
} ).start();
}
}
1.3 Lambda 表达式的标准格式
- 匿名内部类中重写run()方法的代码分析:
- run()方法的形式参数为空,说明调用方法时不需要传递参数
- 方法返回值类型为 void,说明方法执行没有结果返回
- 方法体中的内容,是我们具体要做的事情
- Lambda 表达式的代码分析:
- ( ):里面没有内容,可以看成是方法形式参数为空
- ->:用箭头指向后面要做的事情
- { }:包含一段代码,称之为代码块,可以看成是方法体中的内容
-
Lambda表达式的三要素:
- 形式参数
- 箭头
- 代码块
-
Lambda 表达式的标准格式:
- 格式:** (形式参数) -> {代码块}**
- 形式参数:多个参数用逗号隔开,没有参数留空即可
- ->:由英文中画线和大于符号组成,固定写法。代表指向动作
- 代码块:是我们具体要做的事情,也就是以前方法体中的内容
1.4 Lambda 表达式的练习
- Lambda表达式的使用前提:
- 有一个接口
- 接口中有且仅有一个抽象方法
1.4.1 练习1(抽象方法无参无返回值)
- 定义一个接口(Eatable),里面有一个抽象方法:void eat();
- 定义一个测试类(EatableDemo),在测试类中提供两个方法:
useEatable(Eatable e)
main(),在主方法中调用useEatable方法
package FH.Lambda02;
public interface Eatable {
void eat();
}
package FH.Lambda02;
public class EatableImpl implements Eatable{
@Override
public void eat() {
System.out.println("干饭啦!!!");
}
}
package FH.Lambda02;
/*
练习1:
定义一个接口(Eatable),里面有一个抽象方法:void eat();
定义一个测试类(EatableDemo),在测试类中提供两个方法:
useEatable(Eatable e)
main(),在主方法中调用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();
}
}
1.4.2 练习2(抽象方法带参无返回值)
- 定义一个接口(Flyable),里面有一个抽象方法:void fly(String s);
- 定义一个测试类(FlyableDemo),在测试类中提供两个方法:
useFlyable(Flyable f)
main(),在主方法中调用useFlyable方法
package FH.Lambda03;
public interface Flyable {
void fly(String s);
}
package FH.Lambda03;
/*
练习2:
定义一个接口(Flyable),里面有一个抽象方法:void fly(String s);
定义一个测试类(FlyableDemo),在测试类中提供两个方法:
useFlyable(Flyable f)
main(),在主方法中调用useFlyable方法
*/
public class FlyableDemo {
public static void main(String[] args){
useFlyable(new Flyable() {
@Override
public void fly(String s) {
System.out.println(s);
}
});
useFlyable((String s) -> {
System.out.println(s);
});
}
private static void useFlyable(Flyable f) {
f.fly("半仙儿");
}
}
1.4.3 练习3(抽象方法带参带返回值)
- 定义一个接口(Addable),里面有一个抽象方法:int add(int x,int y);
- 定义一个测试类(AddableDemo),在测试类中提供两个方法:
useAddable(Addable a)
main(),在主方法中调用useAddable方法
package FH.Lambda04;
public interface Addable {
int add(int x,int y);
}
package FH.Lambda04;
/*
练习3:
定义一个接口(Addable),里面有一个抽象方法:int add(int x,int y);
定义一个测试类(AddableDemo),在测试类中提供两个方法:
useAddable(Addable a)
main(),在主方法中调用useAddable方法
*/
public class AddableDemo {
public static void main(String[] args){
useAddable(new Addable() {
@Override
public int add(int x, int y) {
return x+y;
}
});
useAddable((int x,int y) -> {
return x + y;
});
}
private static void useAddable(Addable a){
int sum = a.add(367,376);
System.out.println(sum);
}
}
1.5 Lambda 表达式的省略模式
Lambda 表达式的省略规则:
- 参数类型可以省略,多个参数的情况下不能只省略一个
- 如果参数有且仅有一个,小括号可以省略
- 如果代码块的语句只有一条,可以省略大括号和分号,如果有return,return也要省略
package FH.Lambda05;
/*
Lambda表达式的省略模式
*/
public class LambdaDemo {
public static void main(String[] args){
useAddable((int x,int y) -> {
return x + y;
});
//1.参数的类型可以省略
useAddable((x,y) -> {
return x + y;
});
useFlyable((s) -> {
System.out.println(s);
});
//2.如果参数有且仅有一个,那么小括号可以省略
useFlyable(s -> {
System.out.println(s);
});
//如果代码块的语句只有一条,可以省略大括号和分号
useFlyable(s -> System.out.println(s));
//3.如果代码块的语句只有一条,可以省略大括号和分号,如果有return,return也要省略
useAddable((x,y) -> x + y);
}
private static void useFlyable(Flyable f){
f.fly("半仙儿");
}
private static void useAddable(Addable a){
int sum = a.add(367, 376);
System.out.println(sum);
}
}
1.6 Lambda 表达式的注意事项
- 使用Lambda必须要有接口,且接口中有且仅有一个抽象方法
- 必须有上下文环境,才能推导出Lambda对应的接口
- 根据 局部变量的赋值 得知Lambda对应的接口:Runnable r = () -> System.out.println(“半仙儿努力考研”);
- 根据 调用方法的参数 得知Lambda对应的接口:new Thread(() -> System.out.println(“半仙儿努力考研”)).start();
package FH.Lambda06;
public interface Inter {
void show();
}
package FH.Lambda06;
public class LambdaDemo {
public static void main(String[] args){
//使用Lambda表达式必须要有接口,并且接口中有且仅有一个抽象方法
useInter(() -> System.out.println("半仙儿努力考研"));
//必须有上下文环境,才能推导出Lambda对应的接口
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("半仙儿努力考研");
}
}).start();
// () -> System.out.println("半仙儿努力考研");//单独的Lambda没有意义
Runnable r = () -> System.out.println("半仙儿努力考研");
new Thread(r).start();
new Thread(() -> System.out.println("半仙儿努力考研")).start();
}
private static void useInter(Inter i){
i.show();
}
}
1.7 Lambda 表达式和匿名内部类的区别
- 所需类型不同
- 匿名内部类:可以是接口,抽象类,具体类
- Lambda表达式:只能是接口
- 使用限制不同
- 如果接口中有且仅有一个抽象方法,可以使用Lambda表达式,也可以使用匿名内部类
- 如果接口中多于一个抽象方法,只能使用匿名内部类
- 实现原理不同
- 匿名内部类:编译之后,产生一个单独的 .class字节码文件
- Lambda表达式:编译之后,没有一个单独的 .class字节码文件,对应字节码会在运行的时候动态生成
package FH.Lambda07;
public interface Inter {
void show();
void show2();
}
package FH.Lambda07;
public abstract class Animal {
public abstract void method();
}
package FH.Lambda07;
public class Student {
public void study(){
System.out.println("半仙儿努力考研");
}
}
package FH.Lambda07;
public class LambdaDemo {
public static void main(String[] args){
//匿名内部类
/* useInter(new Inter() {
@Override
public void show() {
System.out.println("接口");
}
});
useAnimal(new Animal() {
@Override
public void method() {
System.out.println("抽象类");
}
});
useStudent(new Student(){
@Override
public void study() {
System.out.println("具体类");
}
});
*/
//Lambda
// useInter(() -> System.out.println("接口"));
// useAnimal(() -> System.out.println("抽象类"));
// useStudent(() -> System.out.println("具体类"));
useInter(new Inter() {
@Override
public void show() {
System.out.println("show");
}
@Override
public void show2() {
System.out.println("show2");
}
});
}
private static void useStudent(Student s){
s.study();
}
private static void useAnimal(Animal a){
a.method();
}
private static void useInter(Inter i){
i.show();
}
}