若定义为 final,则 Java 编译器则会在内部类生成一个外部变量的拷贝,而且既可以保证内部类可以引用外部属性,又能保证值的唯一性。
若不定义为 final,则无法通过编译(jdk 1.6 测试过)。因为编译器不会给非 final 变量进行拷贝,那么内部类引用的变量就是非法的。
匿名内部类用法:
public class TryUsingAnonymousClass {
public void useMyInterface() {
final Integer number = 123;
System.out.println(number);
MyInterface myInterface = new MyInterface() {
@Override
public void doSomething() {
System.out.println(number);
}
};
myInterface.doSomething();
System.out.println(number);
}
}
编译后的结果
class TryUsingAnonymousClass$1
implements MyInterface {
private final TryUsingAnonymousClass this$0;
private final Integer paramInteger;
TryUsingAnonymousClass$1(TryUsingAnonymousClass this$0, Integer paramInteger) {
this.this$0 = this$0;
this.paramInteger = paramInteger;
}
public void doSomething() {
System.out.println(this.paramInteger);
}
}
因为匿名内部类最终会编译成一个单独的类,而被该类使用的变量会以构造函数参数的形式传递给该类,例如:Integer paramInteger
,如果变量不定义成 final 的,paramInteger 在匿名内部类被可以被修改,进而造成和外部的 paramInteger 不一致的问题(因为逻辑的意图就是为了引用外层那个特定的变量对应的值或对象),为了避免这种不一致的情况,因次 Java 规定匿名内部类只能访问 final 修饰的外部变量。
需要注意的是,在 Kotlin 的 lambda 表达式中,引用外层方法的局部变量时,虽然该变量可以不标记为 final,但实际上是 final 的,在 lambda 表达式内变更会报错。
方法内不允许使用 static 修饰变量。
且对于外部类的成员变量,是可以随意使用的,不需要修饰其为 final,因为匿名内部类会持有外部类的引用,实际上是通过该外部类的引用去使用其成员变量。