可能有些人没遇到过 list.forEach 中变量必须为 final 的问题,那就先举两个例子
示例1,如下:
public static void main(String[] args) {
List<String> stringList = new ArrayList<>();
stringList.add("a");
stringList.add("b");
stringList.add("c");
stringList.add("d");
int i = 0;
stringList.forEach(s -> {
System.out.println(s + i);
});
}
示例2,如下:
public static void main(String[] args) {
List<String> stringList = new ArrayList<>();
stringList.add("a");
stringList.add("b");
stringList.add("c");
stringList.add("d");
int i = 0;
stringList.forEach(s -> {
System.out.println(s + i);
// 示例1与示例2的区别
i = i + 1
});
}
咋一看,两个示例没什么问题,都能正常运行。但是实际运行结果如下。
示例1运行结果
a0
b0
c0
d0
而示例2无法运行,编辑器给了如下提示
Error:(16,36)java:从lambda 表达式引用的本地变量必须是最终变量或实际上的最终变量
要把示例2修正为可以运行的代码,可以做如下修正:
public static void main(String[] args) {
List<String> stringList = new ArrayList<>();
stringList.add("a");
stringList.add("b");
stringList.add("c");
stringList.add("d");
final int[] i = {0};
stringList.forEach(s -> {
System.out.println(s + i[0]);
i[0] = i[0] + 1;
});
}
就以上的现象,用三个问题来简单说明。
问题一,为什么示例2的 int i = 0 必须用 final 修饰?
答:forEach 在此处使用的是 lambda 表达式,可以简单的把 lambda 表达式 理解为匿名内部类(lambda 表达式不仅仅是内部类这么简单)。而匿名内部类的变量必须用 final 修饰。
答:类的生命周期比方法的生命周期长,同理匿名类的生命周期比方法的生命周期长。换句话说,方法运行完了,变量释放了,但是匿名内部类还在。这时就要求匿名内部类引用的变量必须还在,这样才能保持数据的一致性。
问题三,为什么变量 int i 要改为数组 int[] i?
答:因为 final int i 中,i 的值是无法改变的,但是方法中需要一个可以改变的变量。在 final int[] i 中,i 的引用地址是不变的,但是 i 的属性是可以改变的。
问题四,这里抛出一个问题,内部类中的变量和方法的局部变量是同一个变量吗?如果内部类中的变量采用局部变量的副本不就解决生命周期问题了吗?可不可以呢?
答:对于是不是同一个变量我们做个实验


至于可不可以,事实上是不可以的,因为如果使用副本就没办法保证数据一致性了。
注意:java8的局部变量在内部类中使用默认加上了final,不需要手动添加了,有兴趣的小伙伴可以观察一下编译文件。
虽然我还没有找到对于局部变量增加final关键字是否可以延长其生命周期的相关资料,但是通过实验目前我得到的结论应该是可以。
以上只是简单的说明,便于大家理解。大家可以继续深究一下里面的知识点。