局部内部类和匿名内部类访问局部变量时,为什么变量必须加上final?

项目场景:

内部类:

  • 成员内部类
  • 局部内部类
  • 静态内部类
  • 匿名内部类

问题描述

  • 局部内部类和匿名内部类访问局部变量时,为什么变量必须加上final?

局部内部类代码实例:

@Override
    public class OutClass {
        public void outPrint( int x) {
            final int age = 12;
            System.out.println(age);
            class InClass {
                public void InPrint() {
                    System.out.println(age);
                }
            }
            new InClass().InPrint();
        }
    }

	}

这里写了一个outClass类,内部方法中有个InClass内部类,内部类访问了外部类的一个方法中的一个局部变量 age。
在这里,age 不能修改,默认是:final。否则,会报错:

Variable ‘age’ is accessed from within inner class, needs to be final or effectively final


原因分析:

为什么这里的局部变量不能修改?

  • 追究其根本原因就是作用域中变量的生命周期导致的;
  • 首先需要知道的一点是:
    • 内部类和外部类是处于同一个级别的(都是类),内部类不会因为定义在方法中,就会随着方法的执行完毕被销毁。
  • 这样就会产生问题:
    • 当外部类的方法结束时,局部变量就会被销毁了,但是内部类对象可能还存在(只有没有人再引用它时,才会死亡)。
  • 这样就会出现了一个矛盾:
    • 内部类对象访问了一个不存在的变量。
  • 为了解决这个问题,就将局部变量复制了一份作为内部类的成员变量
    • 这样当局部变量死亡后,内部类仍可以访问它,实际访问的是局部变量的"copy"。这样就好像延长了局部变量的生命周期。

可以通过javap 命令,反编译 .class 文件进行实验:

在命令行窗口中先执行命令(编译):

javac OutClass.java

进行编译,会得到两个字节码文件:
OutClass$1InClass.class、OutClass.class:
在这里插入图片描述
使用:javap(Java class 文件分解器)可以反编译,也可以查看java编译器生成的字节码。

  • 这里再执行命令:
 javap -private OutClass$1InClass 

进行反编译, -private 表示显示所有类和成员,执行后会得到如下结果:
在这里插入图片描述
可以看到:final int val$age;作为成员变量出现在了InClass内部类中。得出,上面的结论是正确的:

  • 将局部变量复制了一份作为内部类的成员变量

那这跟final修饰有什么关系呢?

那当然是:为了保证,外部类的局部变量和内部类的成员变量(copy的)的一致性。

  • 只赋一次值,之后不能改(这不就是final的特性嘛,我拿过来直接就用),这样就很便捷的保证了一致性。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

鳄鱼杆

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

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

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

打赏作者

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

抵扣说明:

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

余额充值