1. Lambda表达式的本质
Lambda表达式和匿名内部类,其实都是用来生成一个接口的实例。
与通过类来实现接口,然后类实例化的一个对象相比,Lambda表达式和匿名内部类语法更加简洁,不必去定义类,就可以生成接口的实例。
2. 语法
例如加密这个事情,加密接口如下:
/** * 加密接口 */ public interface IEncoder { /** * 对数字加密 */ public int encode(int num); } 2.1 通过类实现加密 要实现具体的加密算法,最普通的方式就是定义类来实现接口,下面这段代码纯属常规操作,无需解释。 /** * 加法加密 */ public class AddEncoder implements IEncoder { @Override public int encode(int num) { return num + 1; } /** * 测试 */ public static void main(String[] args) { IEncoder encoder = new AddEncoder(); int result = encoder.encode(1); System.out.println("加密结果:" + result); } } 2.2 通过匿名内部类实现加密 如果通过匿名内部类的话,则无需专门定义一个类,直接在需要的时候定义一个匿名的内部类即可。 public class AnonymousDemo { public static void main(String[] args) { // 使用时临时定义类 IEncoder encoder = new IEncoder() { @Override public int encode(int num) { return num + 1; } }; int result = encoder.encode(1); System.out.println("加密结果:" + result); } } 对于上面两个例子中的encoder对象来说,他的能力其实是一样的,都是一个具备加法加密方法的实例。但是在具体产生这个对象的方式上,匿名内部类可以临时起意,进行加密方法的定制,相比与专门定义一个类更加轻量级。 2.3 通过Lambda表达式实现加密 Lambda表达式更加简单,但是本质上依然是生成一个接口实例。 public class LambdaDemo { public static void main(String[] args) { // 通过表达式定义接口的实力 IEncoder encoder = (num) -> { return num + 1; }; int result = encoder.encode(1); System.out.println("加密结果:" + result); } } 可以看到,其实Lambda表达式,还是在表现一个方法的具体实现。 (num)表示参数 ->表示这是一个Lambda表达式 {return num+1;}代码块,编写具体的方法逻辑。 2.4 Lambda表达式的简略写法 当参数列表只有一个参数时,可以省略参数的圆括号。 当代码块中只有一条语句时,可以直接省略代码块的{}。 当代码块只有一个return时,也可以省略。 所以上面的表达式可以简化为: public class LambdaDemo { public static void main(String[] args) { // 通过表达式定义接口的实力 IEncoder encoder = num -> num + 1; int result = encoder.encode(1); System.out.println("加密结果:" + result); } } 3. Lambda表达式存在的意义 匿名内部类和Lambda表示式其实都是为了简化接口实例化的代码。 当接口的实例仅仅是拿过来使用1次时,单独去写一个类有点浪费了,不如直接一个表达式搞定。 Lambda比匿名内部类更加简单,但是Lambda表达式只能面向只有一个方法的接口,而匿名函数可以支持有任意多个方法的接口 4. 函数式接口 Lambda表达式能够支持的,只有一个方法的接口,可以被称为函数式接口。 Java8为函数式接口提供了一个注解@FunctionalInterface,用来检查接口是否为函数式接口,可见函数式接口的重要性! 语法如下: /** * 加密接口 */ @FunctionalInterface public interface IEncoder { /** * 对数字加密 */ public int encode(int num); }