一尘不染

Lambda Java参考

java

我想将转换anonymous classlambda expression。但是这个匿名类我使用this关键字。

例如,我写了这个简单的Observer/Observable模式:

import java.util.ArrayList;
import java.util.Collection;

public static class Observable {
    private final Collection<Observer> notifiables = new ArrayList<>();

    public Observable() { }

    public void addObserver(Observer notifiable) { notifiables.add(notifiable); }
    public void removeObserver(Observer notifiable) { notifiables.add(notifiable); }

    public void change() {
        notifiables.forEach(notifiable -> notifiable.changed(this));
    }
}

public interface Observer {
    void changed(Observable notifier);
}

和带有匿名类的此示例代码(使用this关键字):

public class Main {

    public static void main(String[] args) {
        Observable observable = new Observable();
        observable.addObserver(new Observer() {
            @Override
            public void changed(Observable notifier) {
                notifier.removeObserver(this);
            }
        });
        observable.change();
    }
}

但是当我将其转换为lambda表达式时:

public class Main {

    public static void main(String[] args) {
        Observable observable = new Observable();
        observable.addObserver(notifier -> { notifier.removeObserver(this); });
        observable.change();
    }
}

我收到此编译错误:

Cannot use this in a static context and in a non `static` context



public class Main {
    public void main(String[] args) {
        method();
    }

    private void method() {
        Observable observable = new Observable();
        observable.addObserver(notifier -> {
                notifier.removeObserver(this);
        });
        observable.change();
    }
}

编译错误是:

The method removeObserver(Main.Observer) in the type Main.Observable is not applicable for the arguments (Main)

所以我的问题是:有没有办法引用“ lambda对象” this


阅读 350

收藏
2020-03-25

共1个答案

一尘不染

你不能thislambda表达式中引用。的语义this已更改为仅在lambda中引用周围类的实例。无法this从lambda内部引用lambda表达式。

问题是你this在该main()方法中使用。主要方法是静态的,并且没有对表示的对象的引用this

this内部类的实例内部使用时,你将引用内部类的实例。Lambda表达式不是内部类,this没有引用Lambda表达式的实例。它引用你在其中定义lambda表达式的类的实例。在你的情况下,它将是Main的实例。但是,由于你使用的是静态方法,因此没有实例。

这是你的第二个编译错误告诉你的内容。你将Main实例移交给你的方法。但是你的方法签名需要一个Observer实例。

更新:

在Java语言规范15.27.2说:

与出现在匿名类声明中的代码不同,名称的含义以及出现在lambda正文中的this和super关键字以及引用的声明的可访问性与周围环境相同(除了lambda参数引入新名称)。

在lambda表达式的主体中(无论是隐式的还是隐式的)此透明性(即与周围环境相同)将为实现提供更大的灵活性,并防止主体中不合格名称的含义相互依赖在超载解决方案上。

实际上,lambda表达式需要谈论自己(以递归方式调用其自身或调用其其他方法)是很不寻常的,而使用名称来引用封闭类中的东西则更为常见。否则将被遮盖(这是toString())。如果有必要让lambda表达式引用自身(如通过this),则应改用方法引用或匿名内部类。

2020-03-25