在分析之前,首先说一下内部类定义在局部的特点。
内部类被定义在局部时:
1. 不可以被成员修饰符修饰;
2. 可以直接访问外部类中的成员,因为还持有外部类中的引用;
3. 但是不可以访问它所在局部中的变量,只能访问被final修饰的局部变量。
其中,匿名内部类其实就是一条特殊的语句,被定义在方法中,所以,匿名内部类也属于被定义在局部的内部类,符合上述特点。
在《Java编程思想》中有这样 一句话,“如果定义一个匿名内部类,并且希望它使用一个在其外部定义的对象,那么编译器会要求其参数引用是final的”。
这里所说的“外部定义的对象”,指的是所有外来的对象,包括外部方法的形参、局部变量、基本类型或自定义类型等。
这里之所以只能用final修饰的参数,是变量作用域的原因。虽然匿名内部类被定义在方法内部,但匿名内部类是单独的个体,编译时随外部类一起被编译成为Outer$1.class文件,并不是方法被调用时才会被执行。方法中的局部变量只是在方法被调用时被创建在栈内存中,调用完毕会自动清空栈,所以,匿名内部类要想使用方法内部的变量,只能将该变量用final修饰,即定义为常量。
interface Destination{
String readLabel();
}
public class Parcel{
public Destination destination(final String dest){
return new Destination(){
private String label = dest;
public String readLabel(){
return label;
}
};
}
public static void main(String[] args){
Parcel p = new Parcel();
Destination d = p.destination("hello");
}
}
如上例所示,匿名内部类中用到了函数destination中的参数,就必须将该参数用final修饰,否则,编译会提示错误。
但是,如果你使用的JDK为1.8版本,就算局部变量不用final修饰,编译也会通过,而且可以正常运行。但编译器会默认该变量为final类型。这也许是新版本的一个bug吧。
本文探讨了Java中匿名内部类如何使用外部定义的对象,指出内部类在局部定义时的特性,强调匿名内部类必须访问final修饰的局部变量。引用《Java编程思想》中的观点,解释了为何需要final修饰,原因是匿名内部类在编译时生成独立的字节码文件,而局部变量在方法调用结束后会被清除。在JDK 1.8及更高版本中,即使不使用final修饰,编译器也会默认将其视为final。
175

被折叠的 条评论
为什么被折叠?



