首页 > 编程笔记 > Java笔记

Lambda表达式与匿名内部类的联系和区别

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 = "https://www.xinbaoku.com/";
        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();
    }
}
输出结果为:

url 局部变量为:https://www.xinbaoku.com/
外部类的 age 实例变量为:12
外部类的 name 类变量为:新宝库
8

上面程序使用 Lambda 表达式创建了一个 Displayable 的对象,Lambda 表达式的代码块中的代码第 19、21 和 22 行分别示范了访问“effectively final”的局部变量、外部类的实例变量和类变量。从这点来看, Lambda 表达式的代码块与匿名内部类的方法体是相同的。

与匿名内部类相似的是,由于 Lambda 表达式访问了 url 局部变量,因此该局部变量相当于有一个隐式的 final 修饰,因此同样不允许对 url 局部变量重新赋值。

当程序使用 Lambda 表达式创建了 Displayable 的对象之后,该对象不仅可调用接口中唯一的抽象方法,也可调用接口中的默认方法,如上面程序代码第 26 行所示。

Lambda 表达式与匿名内部类主要存在如下区别。

对于 Lambda 表达式的代码块不允许调用接口中定义的默认方法的限制,可以尝试对上面的 LambdaAndInner.java 程序稍做修改,在 Lambda 表达式的代码块中增加如下一行:

// 尝试调用接口中的默认方法,编译器会报错
System.out.println(add(3, 5));

虽然 Lambda 表达式的目标类型 Displayable 中包含了 add() 方法,但 Lambda 表达式的代码块不允许调用这个方法;如果将上面的 Lambda 表达式改为匿名内部类的写法,当匿名内部类实现 display() 抽象方法时,则完全可以调用这个 add() 方法,如下面代码所示。
public void test() {
    String url = "https://www.xinbaoku.com/";
    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();
}

所有教程

优秀文章