我想将转换anonymous class为lambda expression。但是这个匿名类我使用this关键字。
anonymous class
lambda expression
this
例如,我写了这个简单的Observer/Observable模式:
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?
你不能this在lambda表达式中引用。的语义this已更改为仅在lambda中引用周围类的实例。无法this从lambda内部引用lambda表达式。
lambda
问题是你this在该main()方法中使用。主要方法是静态的,并且没有对表示的对象的引用this。
main()
在this内部类的实例内部使用时,你将引用内部类的实例。Lambda表达式不是内部类,this没有引用Lambda表达式的实例。它引用你在其中定义lambda表达式的类的实例。在你的情况下,它将是Main的实例。但是,由于你使用的是静态方法,因此没有实例。
这是你的第二个编译错误告诉你的内容。你将Main实例移交给你的方法。但是你的方法签名需要一个Observer实例。
更新:
在Java语言规范15.27.2说:
与出现在匿名类声明中的代码不同,名称的含义以及出现在lambda正文中的this和super关键字以及引用的声明的可访问性与周围环境相同(除了lambda参数引入新名称)。
在lambda表达式的主体中(无论是隐式的还是隐式的)此透明性(即与周围环境相同)将为实现提供更大的灵活性,并防止主体中不合格名称的含义相互依赖在超载解决方案上。
实际上,lambda表达式需要谈论自己(以递归方式调用其自身或调用其其他方法)是很不寻常的,而使用名称来引用封闭类中的东西则更为常见。否则将被遮盖(这是toString())。如果有必要让lambda表达式引用自身(如通过this),则应改用方法引用或匿名内部类。