zoukankan      html  css  js  c++  java
  • 简单说明:list.forEach 中变量必须为 final 的问题

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

  • 相关阅读:
    109. 有序链表转换二叉搜索树
    108. 将有序数组转换为二叉搜索树
    235. 二叉搜索树的最近公共祖先
    538. 把二叉搜索树转换为累加树
    230. 二叉搜索树中第K小的元素
    669. 修剪二叉搜索树
    513. 找树左下角的值
    637. 二叉树的层平均值
    671. 二叉树中第二小的节点
    DDL-Oracle中的5种约束总结(未完待续)
  • 原文地址:https://www.cnblogs.com/zhenggc/p/13655489.html
Copyright © 2011-2022 走看看