首先来看,这两者的示例
这样写的点击事件代码,就是使用了匿名内部类,使用匿名内部类来设置点击事件监听器的示例:
imageView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (onAddPhotoClickListener != null) {
onAddPhotoClickListener.onAddPhotoClick();
}
}
});
而这样写的代码,就是使用了lambda表达式来设置点击事件监听器的示例:
imageView.setOnClickListener(v -> {
if (onAddPhotoClickListener != null) {
onAddPhotoClickListener.onAddPhotoClick();
}
});
使用匿名内部类和lambda表达式设置监听器都可以达到相同的功能目的,但它们在语法简洁性、编译效率、可读性以及对作用域变量的访问方面存在一些区别:
语法简洁性
- Lambda表达式提供了一种更简洁、更直观的方式来编写单抽象方法接口的实现。它们减少了模板代码的数量,使代码更加清晰。
- 匿名内部类虽然功能强大,但它们需要更多的模板代码,例如声明新的类、使用关键字
new
等。
编译效率
- 在编译时,lambda表达式和匿名内部类都会被转换成类的私有静态方法。在Java 8及以上版本,lambda表达式通过
invokedynamic
指令支持更高效的执行方式,而匿名内部类则转换成常规的类文件。 - 对于lambda表达式,Java虚拟机(JVM)可以利用
invokedynamic
指令的延迟绑定特性,提高性能。
可读性
- Lambda表达式通常提高了代码的可读性,特别是当实现非常简短时(例如单行实现)。它们让代码更加专注于做什么,而不是如何做。
- 匿名内部类在实现非常复杂或需要多个方法时可能更有优势,但对于单方法接口,它们的冗长可能会影响代码的可读性。
对作用域变量的访问
- Lambda表达式可以捕获并使用
final
或事实上的final
(effectively final,即没有被后续修改的变量)局部变量。这与匿名内部类类似,但lambda表达式的语法更加简洁。 - 匿名内部类也可以访问
final
或事实上的final
局部变量,但是在使用时可能会更加繁琐,尤其是当需要访问外部类的成员时。
结论
选择使用lambda表达式还是匿名内部类主要取决于你的具体需求、代码的目标可读性以及你的工作环境。对于单抽象方法接口的实现,lambda表达式通常是更现代、更优雅的选择,特别是在Java 8及以上版本中。然而,在某些情况下,特别是当需要实现多个方法或者编写更复杂的逻辑时,匿名内部类可能更合适。