为什么局部内部类和匿名内部类只能访问局部final变量

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,对他初始化后,就不再去修改这个变量,就保证了内部类的成员变量和方法的局部变量的一致性。这实际上也是一种妥协。使得局部变量和内部类简历的拷贝保持一致。

posted @ 2021-05-19 21:05  刘指导  阅读(97)  评论(0编辑  收藏  举报