之所以只能访问final局部变量,主要是因为内部类和局部变量生命周期的不同。内部类和外部类一样,本质上是一个类,只有在没有变量引用的情况下内部类才会被销毁。而局部变量在方法结束后就会被销毁,这会导致内部类使用一个已被销毁的变量。为了解决这个问题,编译器会将这个局部变量拷贝成为局部类的成员变量。
public class Test {
public static void main(String[] args) {
}
public void test(final int a) {
new Thread(){
public void run() {
System.out.println(a);
};
}.start();
}
}
这段代码会被编译成两个class文件:Test.class和TestX.class(X为正整数)。对TestX.class进行反汇编可以发现,其构造器有两个参数,一个是指向外部类对象的引用,一个是int型变量。实际上,如果局部变量的值在编译期间就可以确定,则直接在匿名内部里面创建一个拷贝。如果局部变量的值无法在编译期间确定,则通过构造器传参的方式来对拷贝进行初始化赋值。