为什么局部内部类只能访问方法中final类型的局部变量?

来源:互联网 发布:mac airplay无法打开 编辑:程序博客网 时间:2024/06/05 07:01
public class OuterClass2 {    static int i =1;    int ii=1;    public OuterClass2() {        // TODO Auto-generated constructor stub    }    private class InnerClass implements OutInterface{        private int i2=2;        public void my(final String x,final int e,String xx){            i2++;            final String m="a";            OuterClass2.i++;            OuterClass2.this.ii++;            new OuterClass2().ii++;            class InnerClass3{                InnerClass3(String s,int t){//                  s=xx;//错误                    s=x;                    s=m;                    t=e;                    int a = i;                    int aa = ii;                }            }        }    }}

以上代码可以看出,局部内部类InnerClass3可以访问OuterClass2 中的非final类型的变量,却不能访问定义InnerClass3的方法my()中的非final变量。

原因如下:
一, 当方法被调用运行完毕之后,局部变量就已消亡了。但内部类对象可能还存在,
直到没有被引用时才会消亡。此时就会出现一种情况,就是内部类要访问一个不存在的局部变量。

二,解决这一问题的办法就是使用final修饰局部变量,通过将final局部变量”复制”一份,
复制品直接作为方法内部类中的数据成员,这事方法内部类访问的其实是这个局部变量的复制品!
而且,由于被final修饰的变量赋值后不能再修改,所以就保证了复制品与原始变量的一致。

三,原因二的功能能实现的原因是:Java采用了一种copy local variable(复制局部变量)的方式来实现,
也就是说把定义为final的局部变量拷贝过来用,而引用的也可以拿过来用,只是不能重新赋值。
从而造成了可以access local variable(访问局部变量)的假象,而这个时候由于不能重新赋值,
所以一般不会造成不可预料的事情发生。

四, 使用final修饰符不仅会保持对象的引用不会改变,
而且编译器还会持续维护这个对象在回调方法中的生命周期.
所以这才是final变量和final参数的根本意义.

0 0
原创粉丝点击