关于Lamdba表达式中的局部变量问题
Lamdba表达式中可以使用外层作用域中定义的变量,就像匿名内部类。
如下:
int portNumber = 1337;
Runnable r = () -> System.out.println(portNumber);
但是Lambda对于变量有一些限制,Lamdba可以没有限制的使用实例变量和静态变量,但是局部变量必须显示声明为final或者事实上是final(即一个变量虽然没被final修饰,但是他在后面也没被修改)。
举例:
int portNumber = 1337;
Runnable r = () -> System.out.println(portNumber);
// 编译错误,因为portNumber 变量被赋值2次。
// Lambda表达式引用的局部变量必须是final或者事实上是final
portNumber = 31337;
为什么对局部变量的限制?
首先,实例变量和局部变量背后的实现有一个关键的不同,实例变量存储在堆上,而局部变量保存在栈上。
这种限制存在的原因在于局部变量保存在栈上,并且隐式表示它们仅限于其所在线程。如果允许捕获可改变的局部变量,就会引发造成线程不安全的新的可能性,而这是我们不想看到的(实例变量可以,因为它们保存在堆中,而堆是在线程之间共享的)。