Variable used in lambda expression should be final or effectively final 问题的解决和追根溯源

在这里插入图片描述
翻译:lambda表达式中的变量必须是final的

为什么?

因为lambda表达式和匿名内部类一样,不需要额外定义类就可以直接写接口方法的实现,但是实际上只是简化了我们的写法,还是会由产生的一个内部类做了该接口的方法实现,那么这个时候,可以理解为其实有两个类,外部类的方法,调用了内部类的方法。
而外面那个username,就是外部类的方法中的变量。基于这点认识,我们分析会出现如下两个问题:
(1)如果像上面例子的异步执行,有可能就会出现外部类的方法执行完成出栈了,方法中的局部变量username被垃圾回收了,内部类方法的还没执行完,这个时候,内部类方法也已经找不到username外部类变量的引用了
(2)内部引用外部的变量,会导致数据不一致问题

针对上面两个问题,底层的编译是如何解决的呢?

首先看一张图,这里按照编辑器提醒,重新在外部定义一个值:finalUserName,同时把username赋值给它
在这里插入图片描述

这样就不会报错了,理由是:

(1)内部类会把访问到的外部变量复制一份,保存到内部类中。注意这里的finalUserName跟username,是两个地址的数据了。不是我们一般的String指向运行时常量池的同一个地址。
(2)为了保证内部类中的那份数据与外部类的数据一致,必须加final,例如上面例子中username=“”,那么finalUsername就是””,runAsync里面的打印也是””,随后就算我把username赋值username=“hello ”,里面的那份finalUserName还是“”,但是这里为啥没有显式加上final的声明呢?这是因为Java8之后的lambda表达式中或者匿名类中的局部变量编译器会自动帮我们加上final,也就是隐式final

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值