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的,即内部类不能改变要访问的局部变量的值,这样就不会出现数据不同步的问题了
本文解析了局部内部类如何访问外部方法的局部变量,强调了final局部变量的作用,并讨论了数据同步问题。通过实例和内存示意图,讲解了Java避免生命周期冲突和数据不一致性的机制。
504

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



