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