Lambda只能访问effectively final变量,因编译后会捕获局部变量的副本,为避免数据不一致,要求变量初始化后不可变;可访问final或未被重新赋值的局部变量、实例/静态字段、方法参数及数组引用(元素可改),但不能修改非final局部变量;可通过AtomicInteger、数组包装或终端操作绕过限制,核心是保证闭包内变量状态的一致性。
Java Lambda表达式只能访问有效 final(effectively final)的局部变量,不能修改它们,也不能访问非 final 且后续被重新赋值的变量。
Lambda 表达式在编译后会被转换为合成方法或内部类实例,捕获的局部变量实际是“拷贝”到堆上的闭包中。为避免数据不一致,JVM 要求这些变量在初始化后不能再被修改——即逻辑上等价于 final,哪怕没显式写 final 关键字。
想在 Lambda 中“修改”局部变量?直接赋值会编译报错:
❌ 错误示例:
int count = 0;
list.forEach(x -> count++); // 编译失败:local variables referenced from a lambda expression must be final or effectively final
✅ 替代方案:
Lambda 可以自由读写当前对象的实例变量和调用实例方法,因为它隐式持有对 this 的引用。注意:若在 Lambda 中调用可变实例字段,需自行保证线程安全。
基本上就这些。核心就一条:Lambda 捕获的是变量的“快照”,不是实时引用,所以必须确保它不会被意外改变。