zoukankan      html  css  js  c++  java
  • Java学习笔记23---内部类之局部内部类只能访问final的局部变量

    局部内部类是定义在方法体或代码块中的类,在笔记19中已有过简单介绍。

    今天要讨论的是局部内部类为什么只能访问为常量的局部变量。

    作者: 博客园--蝉蝉

    请尊重作者劳动成果,转载请在标题注明“转载”字样,并标明原文链接:

     http://www.cnblogs.com/chanchan/p/8402411.html 

    参考资料:

    http://www.cnblogs.com/dolphin0520/p/3811445.html

    1.首先来了解一下局部内部类是如何访问局部变量的

    Person类是外部类,LoInClassIntf是接口,localInClassRe是Person类的成员方法,且返回值类型为LoInClassIntf;

    方法内定义了一个局部内部类LoInnerClass,该内部类实现了接口LoInClassIntf;

    方法内还定义了一个final的局部变量a,定义了一个LoInnerClass类型的对象引用loInC;

    代码如下:

     1     //笔记23:内部类--局部内部类--实现接口,返回内部类对象
     2     //接口
     3     public interface LoInClassIntf {
     4         void test();
     5     }
     6 
     7     //方法localInClassRe,返回值为LoInClassIntf,局部内部类来实现该接口,向上转型
     8     public LoInClassIntf localInClassRe() {  
     9         final int a = 1; //常量
    10         
    11         //笔记23--内部类--局部内部类--实现接口
    12         class LoInnerClass implements LoInClassIntf {
    13             public void test() {
    14                 System.out.println("variable a:" + a);
    15             }
    16         }
    17 
    18         LoInnerClass loInC = new LoInnerClass();
    19         return loInC;
    20     }
    21 
    22     public static void main(String[] args) {
    23         //笔记23--局部内部类
    24         Person per = new Person();
    25         
    26         LoInClassIntf lInCIntf = per.localInClassRe();
    27         lInCIntf.test();
    28     }

    输出结果为:

    1 variable a:1

    成员方法localInClassRe执行中的内存示意图如下:

    成员方法localInClassRe执行后的内存示意图如下:

    分析:

        成员方法localInClassRe执行后,方法体中的局部变量a和对象引用loInC都被释放掉了,但分配在堆中的对象未回收,这时由main方法中的局部变量lInCIntf来指向它

        到这里还没什么问题,但第27行,lInCIntf调用了test方法,test方法中访问到了成员方法localInClassRe中的局部变量a,而a此时已不存在了,所以就会出现错误;

        即,局部变量与局部内部类的对象的生命周期不同;

        为解决这一问题,Java把局部内部类要访问的局部变量重新拷贝了一份,并把备份放在内部类的常量池中,这样不论方法有没有执行结束,拷贝都是存在的,就不会再出现访问不存在的变量的错误了。

    成员方法localInClassRe执行中的内存示意图如下:

    成员方法localInClassRe执行后的内存示意图如下:

     上面涉及到的局部变量a是方法体内定义的,如果局部内部类访问的是方法体的参数呢?

    Java采取的方法是,默认为局部内部类的构造方法传入该参数作为构造方法的参数,然后用该参数来初始化内部类中拷贝的变量a。

    局部内部类如何访问局部变量的问题解决了,那么为什么只能访问final的局部变量呢?

    2.数据同步的问题

    上面通过拷贝一份局部变量来解决生命周期不同的问题,如果方法体和局部内部类都改变了a的值会怎么样呢?

    如下图所示:

     

    这样两个a不一致,就会出现数据不同步,下一步应该用a=2还是a=3呢?

    为解决这个问题,Java规定局部内部类可访问的局部变量必须为final的,即内部类不能改变要访问的局部变量的值,这样就不会出现数据不同步的问题了。

  • 相关阅读:
    Google开源单元測试框架Google Test:VS2012 配置
    ubuntu16.04 uninstall cuda 9.0 completely and install 8.0 instead
    ubuntu 16.04 安装cuda的方法
    ubuntu垃圾文件清理方法
    行人检测资源(下)代码数据
    行人检测资源(上)综述文献
    开源深度学习架构Caffe
    python pip 安装库文件报错:pip install ImportError: No module named _internal
    Canny算子
    vmware中nat模式中使用静态ip后无法上网的问题
  • 原文地址:https://www.cnblogs.com/chanchan/p/8402411.html
Copyright © 2011-2022 走看看