项目场景:
内部类:
- 成员内部类
- 局部内部类
- 静态内部类
- 匿名内部类
问题描述
- 局部内部类和匿名内部类访问局部变量时,为什么变量必须加上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的特性嘛,我拿过来直接就用),这样就很便捷的保证了一致性。