Variable used in lambda expression should be final or effectively final


场景描述

我们在使用Java8 lambda表达式的时候时不时会遇到这样的编译报错:
场景描述
这句话的意思是,lambda 表达式中使用的变量应该是 final 或者有效的 final,为什么会有这种规定?


匿名类中的局部变量

其实在 Java 8 之前,匿名类中如果要访问局部变量的话,那个局部变量必须显式的声明为 final,如下代码在 Java 7 中是编译不过的:

    @Test
    public void demo() {
        String version = "1.8";
        foo(new Supplier() {
            @Override
            public String get() {
                return version; // 编译报错 Variable 'version' is accessed from within inner class, needs to be declared final
            }
        });
    }
    private void foo(Supplier supplier) {
        System.out.println(supplier.get());
    }

Java 7 要求 version 这个局部变量必须是 final 类型的,否则在匿名类中不可引用。

我们知道,lambda 表达式是由匿名内部类演变过来的,它们的作用都是实现接口方法,于是类比匿名内部类,lambda 表达式中使用的变量也需要是 final 类型。也就是说我们一开始图片中,i 这个变量需要声明为 final 类型,但是又发现个现象,如图:
现象描述
i 这个变量赋值给了 finalI 变量,但是 finalI 并没有声明为 final 类型,然而代码却能够编译通过,这是因为 Java 8 之后,在匿名类或 Lambda 表达式中访问的局部变量,如果不是 final 类型的话,编译器自动加上 final 修饰符,即 Java8 新特性:effectively final。


思考

前面一直说 Lambda 表达式或者匿名内部类不能访问非 final 的局部变量,这是为什么呢?

  • 首先思考外部的局部变量 finalI 和匿名内部类里面的 finalI 是否是同一个变量?

我们知道,每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接,方法出口等信息,每个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程(《深入理解Java虚拟机》第2.2.2节 Java虚拟机栈)。

就是说在执行方法的时候,局部变量会保存在栈中,方法结束局部变量也会出栈,随后会被垃圾回收掉,而此时,内部类对象可能还存在,如果内部类对象这时直接去访问局部变量的话就会出问题,因为外部局部变量已经被回收了,解决办法就是把匿名内部类要访问的局部变量复制一份作为内部类对象的成员变量,查阅资料或者通过反编译工具对代码进行反编译会发现,底层确实定义了一个新的变量,通过内部类构造函数将外部变量复制给内部类变量。

  • 为何还需要用final修饰?

其实复制变量的方式会造成一个数据不一致的问题,在执行方法的时候局部变量的值改变了却无法通知匿名内部类的变量,随着程序的运行,就会导致程序运行的结果与预期不同,于是使用final修饰这个变量,使它成为一个常量,这样就保证了数据的一致性。


更多知识

了解关于更多 jvm 知识,请参考:深入理解 Java 虚拟机

  • 146
    点赞
  • 293
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 25
    评论
"variable used in lambda expression should be final or effectively final"这个报错是由于在Lambda表达式中使用的变量需要被声明为final或者有效地finalLambda表达式是由匿名内部类演变而来的,而匿名内部类中使用的变量也需要是final类型。这是因为Lambda表达式可以访问外部作用域中的变量,但是不能修改这些变量的值,以确保程序的正确性和可靠性。因此,编译器会对没有被声明为final或有效final的变量进行警告。这个限制的目的是为了避免在Lambda表达式中对外部变量进行修改而引发的潜在问题。通过将变量声明为final或有效final,可以确保在Lambda表达式中使用这些变量时的一致性和可预测性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [编译器说 Lambda 表达式中的变量必须是 final 的,我偏不信](https://download.csdn.net/download/weixin_38651286/13755089)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [【Java 錯誤】Variable used in lambda expression should be final or effectively final](https://blog.csdn.net/qq_45802080/article/details/125451124)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 25
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

发飙的蜗牛咻咻咻~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值