前几天发现以前在自学过程中没有了解到的内容,Lamdba表达式,通过学习发现Lamdba虽然有一些特定的限制,但是在使用中可以让我们的代码更加简洁,也是非常好玩儿的。下面是我了解到的一些关于Lambda表达式的内容,特此记录
在介绍Lambda表达式之前先简单说一下 函数式编程思想
在数学中,函数就是有输入量、输出量的一套计算方案,也就是“拿数据做操作”
面向对象思想强调“必须通过对象的形式来做事情”
函数式思想则尽量忽略面向对象的复杂语法:“强调做什么,而不是以什么形式去做”
而下面要介绍的Lambda表达式就是函数式思想的体现
- Lambda 表达式,也可称为闭包,它是推动 Java 8 发布的最重要新特性。
- Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中)。
- 使用 Lambda 表达式可以使代码变的更加简洁紧凑。
Lambda表达式的标准格式
(形式参数) -> {代码块}
-
基本格式
- 形式参数: 如果有多个参数,参数之间用逗号隔开;如果没有参数,留空即可
- -> : 由英文中画线和大于符号组成,固定写法。代表指向动作
- 代码块: 是我们具体要做的事情,也就是以前我们写的方法体内容
-
组成Lambda表达式的三要素
- 形式参数
- 箭头
- 代码块
-
Lambda表达式的使用前提
- 有一个接口
- 接口中有且仅有一个抽象方法
语法
Lambda表达式的基本语法如下:
(ParameterType parameter) -> {StatementBody}
Lambda表达式的重要特性:
- 参数类型可以省略: 有参数时参数类型可以省略不写,编译器可以统一识别;空参时小括号不可省略。(另,在同一表达式中要么省略全部参数类型,要么全部不省略)
- 参数的小括号可以省略: 当表达式中只有一个参数时可以省略小括号,但有多个参数时必须用小括号括起来
- 代码块的大括号可以省略: 当表达式中的语句体只有一句时,大括号可以省略不写
- return关键字可以省略: 当表达式的语句体只有返回语句的时候,可以省略return关键字,只需要指定表达式的返回值或需要返回的表达式
实例
简单表达式示例:
// 空参,无返回值,控制台打印“Hello World”
() -> {System.out.Println("Hello World")}
// 省略大括号后为:
() -> System.out.Println("Hello World")
// 空参,返回值为0
() -> {return 0}
// 省略大括号和return关键字后为:
() -> 0
// 接收一个int类型的参数,返回其二倍数值
(int n) -> {return n * 2;}
// 省略大、小括号以及参数类型和return关键字后为:
n -> n * 2
// 接收两个int类型的参数,返回其和
(int a, int b) -> {return a + b;}
// 省略大、小括号以及参数类型和return关键字后为:
(a, b) -> a + b
// 接收一个字符串类型并将其打印在控制台,无返回值
(String s) -> {System.out.println(s);}
// 省略大、小括号以及参数类型后为
s -> System.out.println(s)
代码示例:
// 游泳接口
interface Swimming {
void swim();
}
public class TestSwimming {
public static void main(String[] args) {
// 通过匿名内部类实现
goSwimming(new Swimming() {
@Override
public void swim() {
System.out.println("铁汁, 我们去游泳吧");
}
});
// 通过Lambda表达式实现
// 理解: 对于Lambda表达式, 对匿名内部类进行了优化
goSwimming(() -> System.out.println("铁汁, 我们去游泳吧"));
}
// 使用接口的方法(参数为Swimming接口类型)
public static void goSwimming(Swimming swimming) {
swimming.swim();
}
}
Lambda表达式练习1【应用】
-
练习描述
无参无返回值抽象方法的练习
-
操作步骤
- 定义一个接口(Eatable),里面定义一个抽象方法:void eat();
- 定义一个测试类(EatableDemo),在测试类中提供两个方法
- 一个方法是:useEatable(Eatable e)
- 一个方法是主方法,在主方法中调用useEatable方法
-
示例代码
// 接口 public interface Eatable { void eat(); } // 实现类 public class EatableImpl implements Eatable { @Override public void eat() { System.out.println("一天一苹果,医生远离我"); } } // 测试类 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("一天一苹果,医生远离我"); }); // 省略大括号后为(此时小括号不可省略) useEatable(() -> System.out.println("一天一苹果,医生远离我")); } private static void useEatable(Eatable e) { e.eat(); } }
Lambda表达式练习2【应用】
-
练习描述
有参无返回值抽象方法的练习
-
操作步骤
- 定义一个接口(Flyable),里面定义一个抽象方法:void fly(String s);
- 定义一个测试类(FlyableDemo),在测试类中提供两个方法
- 一个方法是:useFlyable(Flyable f)
- 一个方法是主方法,在主方法中调用useFlyable方法
-
示例代码
public interface Flyable { void fly(String s); } 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("飞机自驾游"); }); // 省略小括号和参数类型后为(此时大括号不可省略) useFlyable(s -> { System.out.println(s); System.out.println("飞机自驾游"); }); } private static void useFlyable(Flyable f) { f.fly("风和日丽,晴空万里"); } }
Lambda表达式练习3【应用】
-
练习描述
有参有返回值抽象方法的练习
-
操作步骤
- 定义一个接口(Addable),里面定义一个抽象方法:int add(int x,int y);
- 定义一个测试类(AddableDemo),在测试类中提供两个方法
- 一个方法是:useAddable(Addable a)
- 一个方法是主方法,在主方法中调用useAddable方法
-
示例代码
public interface Addable { int add(int x,int y); } public class AddableDemo { public static void main(String[] args) { //在主方法中调用useAddable方法 useAddable((int x,int y) -> { return x + y; }); // 省略大、小括号,参数类型以及return关键字后为 useAddable((x, y) -> x + y); } private static void useAddable(Addable a) { int sum = a.add(10, 20); System.out.println(sum); } }
Lambda表达式的省略模式【具体】
-
省略的规则
- 参数类型可以省略。但是有多个参数的情况下,不能只省略一个
- 如果参数有且仅有一个,那么小括号可以省略
- 如果代码块的语句只有一条,可以省略大括号和分号,和return关键字
-
代码演示
public interface Addable { int add(int x, int y); } public interface Flyable { void fly(String s); } public class LambdaDemo { public static void main(String[] args) { useAddable((int x,int y) -> { return x + y; }); //参数的类型可以省略 useAddable((x, y) -> { return x + y; }); useFlyable((String s) -> { System.out.println(s); }); //如果参数有且仅有一个,那么小括号可以省略 useFlyable(s -> { System.out.println(s); }); //如果代码块的语句只有一条,可以省略大括号和分号 useFlyable(s -> System.out.println(s)); //如果代码块的语句只有一条,可以省略大括号和分号,如果有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(10, 20); System.out.println(sum); } }
Lambda表达式和匿名内部类的区别
- 所需类型不同
- 匿名内部类:可以是接口,也可以是抽象类,还可以是具体类
- Lambda表达式:只能是接口
- 使用限制不同
- 如果接口中有且仅有一个抽象方法,可以使用Lambda表达式,也可以使用匿名内部类
- 如果接口中多于一个抽象方法,只能使用匿名内部类,而不能使用Lambda表达式
- 实现原理不同
- 匿名内部类:编译之后,产生一个单独的.class字节码文件
- Lambda表达式:编译之后,没有一个单独的.class字节码文件。对应的字节码会在运行的时候动态生成
- 匿名内部类:编译之后,产生一个单独的.class字节码文件
以上内容为我目前所理解的,如有不足(或错误),欢迎补充(或指正)