在 Java 8 之前,匿名内部类在使用外部成员的时候,会报错并提示
“Cannot refer to a non-final variable arg inside an inner class defined in a different method”:
但是在 Java 8 之后,类似场景却没有再提示了:
难道是此类变量可以随便改动了吗?当然不是,当你试图修改这些变量的时候,仍然会提示错误:
可以看到,当试图修改基本数据类型的变量时,编译器的警告变成了
`“Varible 'num' is accessed from within inner class, need to be final or effectively final”
很遗憾,仍然不能修改。
原因分析:
从表面上当然看不出什么原因,看看编译器做了什么工作吧!运行 javac 命令后生成了几个 .class 文件:
不难推断,这个 TestInnerClass$1.class 就是匿名内部类编译后的文件,看看它反编译后是什么内容:
class TestInnerClass$1 extends InnerClass {
TestInnerClass$1(TestInnerClass var1, int var2, DataBean var3) {
super(var1);
this.this$0 = var1;
this.val$num = var2;
this.val$bean = var3;
}
void doSomething() {
super.doSomething();
System.out.println("num = " + this.val$num);
System.out.println("bean name is: " + this.val$bean.name);
}
}
原来,匿名内部类也会被当作普通的类处理,只不过编译器生成它构造方法的时候,除了将外部类的引用传递了过来,还将基本数据类型的变量复制了一份过来,并把引用数据类型的变量引用也传递了过来。因此,基本数据类型的变量当然不能修改了,不然就会跟外部的变量产生不一致,这样的话变量的传递也就变得毫无意义了。
final 关键字除了能让类不能被继承之外,对应到这种场景,就是让变量也不能被重新赋值。