原文链接:http://blog.csdn.net/shouwangsiwang/article/details/8165219
1)所谓“局部内部类”就是在对象的方法成员内部定义的类。而方法中的类,访问同一个方法中的局部变量,是天经地义的。那么为什么要加上一个final呢?
2)原因是:编译程序实现上的困难,难在何处:内部类对象的生命周期会超过局部变量的生命期。为什么?表现在:局部变量的生命期:当该方法被调用时,该方法中的局部变量在栈中被创建(诞生),当方法调用结束时(执行完毕),退栈,这些局部变量全部死亡。而:内部类对象生命期,与其它类一样,当创建一个该局部类对象后,只有没有其它人再引用它时,它才能死亡。完全可能:一个方法已调用结束(局部变量已死亡),但该局部类的对象仍然活着。即:局部类的对象生命期会超过局部变量。
3)退一万步:局部类的对象生命期会超过局部变量又怎样?问题的真正核心是:如果:局部内部类的对象访问同一个方法中的局部变量,是天经地义的,那么:只要局部内部类对象还活着,则:栈中的那些它要访问的局部变量就不能“死亡”(否则:它都死了,还访问个什么呢?),这就是说:局部变量的生命期至少等于或大于局部内部类对象的生命期。而:正是这一点是不可能做到的
4)但是从理论上:局部内部类的对象访问同一个方法中的局部变量,是天经地义的。所以:经过努力,达到一个折中结果:即:局部内部类的对象可以访问同一个方法中的局部变量,只要这个变量被定义为final.那么:为什么定义为final变可以呢?定义为final后,编译程序就好实现了:具体实现方法是:将所有的局部内部类对象要访问的final型局部变量,变成该内部类对象中的一个数据成员。这样,即使栈中局部变量(含final)已死亡,但由于它是final,其值永不变,因而局部内部类对象在变量死亡后,照样可以访问final型局部变量。
归纳上述回答的真正核心是:局部内部类对象中包含有要访问的final型局部变量的一个拷贝,成为它的数据成员。因此,正是在这个意义上,final型局部变量的生命期,超过其方法的一次调用。严格来说,方法调用结束,所有的局部变量(含final)全死亡了。但:局部内部类对象中有final型局部变量的拷贝。
其他:
不管对象是不是final,他的生命周期都是 new开始,垃圾回收结束。
不管变量是不是final,他的生命周期都在于{}中。
类对象(class对象)与其它对象不同,类对象的生命周期 开始于类被加到内存中那一刻,结束于垃圾回收。
类变量(static)与类对象的生命周期相同。
- public class LocalInnerClassTest{
- public static void main(String[] args){
- Outer obj=new Outer(); //生成一个外部类对象
- SuperInner si=obj.outer(); //调用外部类中的outer()方法,返回一个SuperInner类型对象赋值给si
- si.m1(); //调用被覆盖的方法m1(),输出:Inner's m1() 20
- }
- }
- /**
- *定义一个接口SuperInner,内部定义一个抽象方法m1(),无返回类型
- */
- interface SuperInner{
- public void m1();
- }
- /**
- *定义一个类Outer,内部只定义一个方法outer(),返回类型为SuperInner
- */
- class Outer{
- public SuperInner outer(){
- int a=10; //方法中定义一个局部变量a,并赋值为10
- final int b=20; //再定义一个final局部变量b,初始化为20
- class Inner implements SuperInner{ //在outer()方法中定义一个局部内部类Inner,实现接口SuperInner
- public void m1(){ //类中只有一个覆盖接口SuperInner的方法m1()
- System.out.println("Inner's m1()"+a); //编译报错
- System.out.println("Inner's m1() "+b); //编译通过,输出:Inner's m1() 20
- }
- }
- return new Inner();
- }
- }
相关推荐
在 m1() 方法中,我们尝试访问局部变量 a 和 b,结果发现访问 b 成功,而访问 a 不成功。这是因为 JVM 只有在变量被 final 修饰时,才会将其复制给局部内部类。 在 Java 中,final 修饰的局部变量是一个常量,它的...
如果这个内部类可以访问非final的局部变量,那么在方法结束后,这些变量应该已经失效,但内部类依然可以访问它们,这就导致了逻辑上的矛盾。 2. **内部类的实现机制**:Java内部类的实现方式使得它们能够访问外部类...
3. **无法被访问控制修饰符修饰**: 局部变量不能有public、private、protected或包访问权限,因为它们的作用域仅限于当前代码块。 4. **没有默认初始化**: 局部变量不会自动初始化,必须明确赋值后才能使用。 理解...
方法内部类对象不能使用该内部类所在方法的非final局部变量。 四、匿名内部类 匿名内部类是指不具有名称的内部类。匿名内部类适合使用场景包括:只用到类的一个实例、类在定义后马上用到、类非常小、给类命名并...
Java编程语言中有三种主要的变量类型:成员变量(也称为实例变量)、类变量(也称为静态变量)和局部变量。理解这些变量之间的区别是学习Java基础的重要部分,这对于编写高效、可维护的代码至关重要。 1. 成员变量...
因此,局部变量不能在方法外部或类的其他地方访问。 3. **静态变量(Static Variables)**: 静态变量是属于类而非类的实例的变量。这意味着所有类的实例共享同一份静态变量的拷贝。无论创建了多少个对象,静态...
本文旨在深入探讨C语言中的局部变量和全局变量,通过具体的例子和详细的解释,帮助读者更好地理解这两种变量类型的特点和使用方法。 #### 变量概述 在C语言中,**变量**是用于存储数据的标识符。变量可以存储各种...
这意味着,无论你在哪个函数中修改了全局变量的值,这个变化都会被其他所有能访问到该变量的函数所看到。然而,全局变量也有其潜在的问题,比如增加了代码的耦合度,可能导致意外的副作用,因此在编写大型程序时应...
例如,在一个If...Then语句或者For循环中声明的局部变量,只能在这些结构内部访问。 四、生命周期 局部变量的生命周期始于声明时,结束于所在过程执行完毕。这意味着每次调用过程时,都会重新创建并初始化局部变量...
这类变量包括静态局部变量和全局变量,它们存储在静态数据区中。静态存储变量的生命周期与程序的生命周期相同,从程序启动到程序结束。 #### 五、静态局部变量 - **存储位置**:静态局部变量存储在静态数据区。 - ...
与全局变量相反,局部变量是在函数内部声明的变量,其作用域仅限于该函数内部。一旦函数执行完毕,局部变量就会被销毁,释放其所占用的内存空间。局部变量的生命周期与其所在函数的执行周期一致,函数调用时创建,...
- 若全局变量仅由单个函数访问,可以将其改为该函数的静态局部变量,同样能降低耦合度。 - 设计和使用访问静态变量的函数时,需考虑重入问题,因为静态变量都位于静态数据存储区,全局可见。 - 需要一个可重入的函数...
局部变量是在方法内部或者某个代码块中声明的变量。它们的作用域仅限于声明它们的方法或代码块。这意味着一旦执行流离开该方法或代码块,这些变量就会被销毁。 ##### 特点: 1. **存储空间**:局部变量的存储空间是...
1. **访问权限**:局部内部类可以访问方法内的局部变量,但这些变量必须声明为final。 2. **实例化**:局部内部类只能在其定义的方法或构造函数内部实例化。 3. **示例**: ```java public void method() { final...
- **局部变量**: 在函数内部定义,作用范围仅限于该函数内部。 - **静态局部变量**: 同样在函数内部定义,但通过添加`static`关键字,使得它在整个程序运行期间都存在。 #### 三、存储方式与生命周期 - **全局变量...
局部内部类访问外边的局部变量 马克-to-win java视频的介绍