public class Main {
public static void main(String[] args) {
OutClass outClas= new OutClass();
outClas.outPrint(2);
H1 t = new H1();
t.test(3);
}
}
class OutClass{
private int age = 12; //访问外部成员变量不需要final
public void outPrint(final int x){ //需要fianl
class InClass{
public void INPrint(){
System.out.println(x);
System.out.println(age);
}
}
new InClass().INPrint();
}
}
class H1{
public void test(final int b){ //需要final
final int a = 10; //需要final
new Thread(){
@Override
public void run() {
System.out.println(a);
System.out.println(b);
}
}.start();
}
}
首先需要知道一点是:内部类和外部类是处于同一级别,内部类不会因为定义在方法中就会随着方法的执行完毕而销毁。
编译之后会生成两个class文件H1.class 和H1$1.class
这里就会产生一个问题:当外部类的方法结束时,局部变量就会被销毁,但是内部类对象可能还存在(因为内部类和外部类属于同一级别)。这里就会出现一个矛盾:内部类对象访问了一个不存在的变量。为了解决这个问题,jvm就将局部变量复制一份作为内部类的成员变量,这样当局部变量销毁之后,内部类仍然可以访问它,实际访问的是局部变量的copy。这样就好像延长了局部变量的生命周期。
将局部变量复制为内部类的成员变量时,必须保证这两个变量时一样的,也就是如果我们在内部类中修改了成员变量,方法中的局部变量也得跟着改变,怎么解决这个问题呢?
就将局部变量设置为final,对他初始化后,就不再去修改这个变量,就保证了内部类的成员变量和方法的局部变量的一致性。这实际上也是一种妥协。使得局部变量和内部类简历的拷贝保持一致。
原文:https://www.cnblogs.com/liuzhidao/p/14787357.html