为什么内部类可以无限制访问外部类变量
public class Outter {
private Inner inner = null;
public Outter() {
//一些操作
}
public class Inner {
public Inner() {}
}
}
编译器在编译过程中除了会生成一个Outer.class
文件外,还会给内部类单独生成一个字节码文件Outer$Inner.class
对Outer$Inner.class
反编译后会发现,编译器会自动给内部类的构造器添加一个参数(一个指向外部类的引用),即使定义的是一个无参的内部类构造器。因此内部类可以无限制的访问外部类的成员和方法,但同时如果外部类没有实例化,内部类的构造器无法对外部类引用进行赋值,因此内部类的实例化必须在外部类实例化之后
为什么局部类和匿名类只可访问final局部变量
之所以只能访问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型变量。实际上,如果局部变量的值在编译期间就可以确定,则直接在匿名内部里面创建一个拷贝。如果局部变量的值无法在编译期间确定,则通过构造器传参的方式来对拷贝进行初始化赋值。