Java Lambda 表达式的一个重要用法是简化某些匿名内部类的写法,因此它可以部分取代匿名内部类的作用。
Lambda 表达式与匿名内部类的相同点如下:
下面程序示范了 Lambda 表达式与匿名内部类的相似之处。
- @FunctionalInterface
- interface Displayable {
- // 定义一个抽象方法和默认方法
- void display();
-
- default int add(int a, int b) {
- return a + b;
- }
- }
-
- public class LambdaAndInner {
- private int age = 12;
- private static String name = "城东书院";
-
- public void test() {
- String url = "http://www.cdsy.xyz/";
- Displayable dis = () -> {
- // 访问的局部变量
- System.out.println("url 局部变量为:" + url);
- // 访问外部类的实例变量和类变量
- System.out.println("外部类的 age 实例变量为:" + age);
- System.out.println("外部类的 name 类变量为:" + name);
- };
- dis.display();
- // 调用dis对象从接口中继承的add()方法
- System.out.println(dis.add(3, 5));
- }
-
- public static void main(String[] args) {
- LambdaAndInner lambda = new LambdaAndInner();
- lambda.test();
- }
- }
输出结果为:
上面程序使用 Lambda 表达式创建了一个 Displayable 的对象,Lambda 表达式的代码块中的代码第 19、21 和 22 行分别示范了访问“effectively final”的局部变量、外部类的实例变量和类变量。从这点来看, Lambda 表达式的代码块与匿名内部类的方法体是相同的。
与匿名内部类相似的是,由于 Lambda 表达式访问了 url 局部变量,因此该局部变量相当于有一个隐式的 final 修饰,因此同样不允许对 url 局部变量重新赋值。
当程序使用 Lambda 表达式创建了 Displayable 的对象之后,该对象不仅可调用接口中唯一的抽象方法,也可调用接口中的默认方法,如上面程序代码第 26 行所示。
Lambda 表达式与匿名内部类主要存在如下区别。
对于 Lambda 表达式的代码块不允许调用接口中定义的默认方法的限制,可以尝试对上面的 LambdaAndInner.java 程序稍做修改,在 Lambda 表达式的代码块中增加如下一行:
虽然 Lambda 表达式的目标类型 Displayable 中包含了 add() 方法,但 Lambda 表达式的代码块不允许调用这个方法;如果将上面的 Lambda 表达式改为匿名内部类的写法,当匿名内部类实现 display() 抽象方法时,则完全可以调用这个 add() 方法,如下面代码所示。
- public void test() {
- String url = "http://www.cdsy.xyz/";
- Displayable dis = new Displayable() {
-
- @Override
- public void display() {
- // 访问的局部变量
- System.out.println("url 局部变量为:" + url);
- // 访问外部类的实例变量和类变量
- System.out.println("外部类的 age 实例变量为:" + age);
- System.out.println("外部类的 name 类变量为:" + name);
- System.out.println(add(3, 5));
- }
- };
- dis.display();
- }