简单说明:list.forEach 中变量必须为 final 的问题

简介: 可能有些人没遇到过 list.forEach 中变量必须为 final 的问题,那就先举两个例子示例1,如下:public static void main(String[] args) { List<String> stringList = new ArrayList<>(); stringList.

可能有些人没遇到过 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 修饰。

问题二,为什么匿名内部类的变量必须用 final 修饰?

答:类的生命周期比方法的生命周期长,同理匿名类的生命周期比方法的生命周期长。换句话说,方法运行完了,变量释放了,但是匿名内部类还在。这时就要求匿名内部类引用的变量必须还在,这样才能保持数据的一致性。

问题三,为什么变量 int i 要改为数组 int[] i?

答:因为 final int i 中,i 的值是无法改变的,但是方法中需要一个可以改变的变量。在 final int[] i 中,i 的引用地址是不变的,但是 i 的属性是可以改变的。

以上只是简单的说明,便于大家理解。大家可以继续深究一下里面的知识点。


如果文章有帮助到了你,欢迎点赞、转发。

如果文章有错误的地方,欢迎留言交流。

image

相关文章
|
8月前
|
存储 安全 Java
java集合框架及其特点(List、Set、Queue、Map)
java集合框架及其特点(List、Set、Queue、Map)
|
7月前
|
Dart
Dart之集合详解(List、Set、Map)
Dart之集合详解(List、Set、Map)
|
7月前
|
存储 安全 Java
Java集合详解:Set, Map, Vector, List的对比与联系
Java集合框架核心包括List、Set、Map和Vector。List允许重复元素,如ArrayList(适合读取)和LinkedList(适合插入删除)。Set不允许重复,有HashSet(无序)和TreeSet(排序)。Map存储键值对,HashMap(无序)和TreeMap(排序)。Vector是线程安全的ArrayList替代品,但在多线程环境下使用。选择集合类型应根据应用场景,如有序、无序、键值对需求及线程安全考虑。
|
8月前
|
存储 安全 Java
Java集合详解(List、Map、Set)
Java集合详解(List、Map、Set)
90 4
list中所有方法的使用
学习List(列表)的使用方法。 /之前不能使用 值=什么 *之后只能使用 值=什么
81 0
|
8月前
|
SQL XML Java
<foreach>元素中collection=list改成collection=array
<foreach>元素中collection=list改成collection=array
List<Map<String, Object>>,Map<String,List<Map<String, Object>>>多方式循环遍历
List<Map<String, Object>>,Map<String,List<Map<String, Object>>>多方式循环遍历
160 0
|
存储 Dart 数据处理
Dart中常用的集合类型List、Set、Map、Queue
Dart中常用的集合类型List、Set、Map、Queue
|
JSON 数据格式
for_forEach_map有什么区别?区别大了
for、forEach、map日常都在用,但是你知道他们有什么区别吗?为什么要有这么多功能相似的东西?性能怎么样?看这里,我告诉你
93 0
|
存储 安全 算法
Java集合详解(List,Set,Map)
集合的背景 在没有集合类之前,实际上在Java语言里已经有一种方法可以存储对象,那就是数组。数组不仅可以存放基本数据类型也可以容纳属于同一种类型的对象。数组的操作是高效率的,但也有缺点。比如数组的长度是不可以变的,数组只能存放同一种类型的对象(或者说对象的引用)。
228 0