JAVA中内部类(匿名内部类)访问的局部变量为什么要用final修饰?
2017-07-14 22:20
363 查看
本文主要记录:在JAVA中,(局部)内部类访问某个局部变量,为什么这个局部变量一定需要用final 关键字修饰?
首先,什么是局部变量?这里的局部是:在方法里面定义的变量。
因此,内部类能够访问某局部变量,说明这个内部类不是在类中定义的内部类,而是在方法中定义的内部类,称之为:局部内部类。
局部变量的作用域:局部变量是在某个方法中定义,当该方法执行完成后,局部变量也就消失了。【局部变量分配在JVM的虚拟机栈中,这部分内存空间随着程序的执行自动回收】,也即:局部变量的作用域是在 “方法的范围内”。
但是,当(局部)内部类访问 局部变量 时,会扩大局部变量的作用域。看下面一个示例:
①第4行在main方法中定义了一个局部变量str,第6行定义了一个局部内部类Thread,并且在局部内部类Thread中访问 str
按理说:当程序执行到第19行时,main()方法就结束了,也即:主线程结束了。局部变量str的生命周期也应该结束了。
但是,Thread线程还未结束,在Thread线程中还能够打印局部str的值。这就表明:局部变量 str 作用域被扩大了。
因此,如果局部变量不用 final 修饰,我们就可以在(局部)内部类中随意修改该局部变量值,而且是在 该局部变量的作用域范围之外可以看到这些修改后的值。这会导致一些问题(我也不知道什么问题,安全性?)
因此,JAVA就规定(局部)内部类访问 的 局部变量必须用 final修饰,以防止更改局部变量的值。
在java中, 方法的内部类可以访问方法中的局部变量,但必须用final修饰才能访问。
直到没有被引用时才会消亡。此时就会出现一种情况,就是内部类要访问一个不存在的局部变量。
二,解决这一问题的办法就是使用final修饰局部变量,通过将final局部变量”复制”一份,
复制品直接作为方法内部类中的数据成员,这事方法内部类访问的其实是这个局部变量的复制品!
而且,由于被final修饰的变量赋值后不能再修改,所以就保证了复制品与原始变量的一致。
三,原因二的功能能实现的原因是:Java采用了一种copy local variable(复制局部变量)的方式来实现,
也就是说把定义为final的局部变量拷贝过来用,而引用的也可以拿过来用,只是不能重新赋值。
从而造成了可以access local variable(访问局部变量)的假象,而这个时候由于不能重新赋值,
所以一般不会造成不可预料的事情发生。
四, 使用final修饰符不仅会保持对象的引用不会改变,
而且编译器还会持续维护这个对象在回调方法中的生命周期.
所以这才是final变量和final参数的根本意义.
转载自:
http://www.cnblogs.com/hapjin/p/5744478.html
http://blog.csdn.net/java20131115/article/details/48346051
首先,什么是局部变量?这里的局部是:在方法里面定义的变量。
因此,内部类能够访问某局部变量,说明这个内部类不是在类中定义的内部类,而是在方法中定义的内部类,称之为:局部内部类。
局部变量的作用域:局部变量是在某个方法中定义,当该方法执行完成后,局部变量也就消失了。【局部变量分配在JVM的虚拟机栈中,这部分内存空间随着程序的执行自动回收】,也即:局部变量的作用域是在 “方法的范围内”。
但是,当(局部)内部类访问 局部变量 时,会扩大局部变量的作用域。看下面一个示例:
public class Test { public static void main(String[] args) { final String str = "hapjin"; new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(50); } catch (InterruptedException e) { e.printStackTrace(); } for(int i = 0; i < 10; i++) System.out.println(str); } }).start(); System.out.println("main thread finished"); } }
①第4行在main方法中定义了一个局部变量str,第6行定义了一个局部内部类Thread,并且在局部内部类Thread中访问 str
按理说:当程序执行到第19行时,main()方法就结束了,也即:主线程结束了。局部变量str的生命周期也应该结束了。
但是,Thread线程还未结束,在Thread线程中还能够打印局部str的值。这就表明:局部变量 str 作用域被扩大了。
因此,如果局部变量不用 final 修饰,我们就可以在(局部)内部类中随意修改该局部变量值,而且是在 该局部变量的作用域范围之外可以看到这些修改后的值。这会导致一些问题(我也不知道什么问题,安全性?)
因此,JAVA就规定(局部)内部类访问 的 局部变量必须用 final修饰,以防止更改局部变量的值。
在java中, 方法的内部类可以访问方法中的局部变量,但必须用final修饰才能访问。
原因:
一, 当方法被调用运行完毕之后,局部变量就已消亡了。但内部类对象可能还存在,直到没有被引用时才会消亡。此时就会出现一种情况,就是内部类要访问一个不存在的局部变量。
二,解决这一问题的办法就是使用final修饰局部变量,通过将final局部变量”复制”一份,
复制品直接作为方法内部类中的数据成员,这事方法内部类访问的其实是这个局部变量的复制品!
而且,由于被final修饰的变量赋值后不能再修改,所以就保证了复制品与原始变量的一致。
三,原因二的功能能实现的原因是:Java采用了一种copy local variable(复制局部变量)的方式来实现,
也就是说把定义为final的局部变量拷贝过来用,而引用的也可以拿过来用,只是不能重新赋值。
从而造成了可以access local variable(访问局部变量)的假象,而这个时候由于不能重新赋值,
所以一般不会造成不可预料的事情发生。
四, 使用final修饰符不仅会保持对象的引用不会改变,
而且编译器还会持续维护这个对象在回调方法中的生命周期.
所以这才是final变量和final参数的根本意义.
转载自:
http://www.cnblogs.com/hapjin/p/5744478.html
http://blog.csdn.net/java20131115/article/details/48346051
相关文章推荐
- JAVA中内部类(匿名内部类)访问的局部变量为什么要用final修饰?
- JAVA中内部类(匿名内部类)访问的局部变量为什么要用final修饰?
- 为什么Java匿名内部类访问的外部局部变量或参数需要被final修饰
- [Java] 匿名内部类访问外部类的局部变量为什么一定得是final类型
- 为什么匿名内部类和局部内部类只能访问被final修饰的局部变量?
- java中内部类访问局部变量的时候,为什么变量必须加上final修饰
- 匿名内部类为什么访问外部类局部变量必须是final的?
- 为什么局部变量必须以final修饰(或者有final实效:java8)才可以在内部类中使用?
- java中内部类访问局部变量为什么要定义局部变量为final
- 局部内部类访问的局部变量不必用final修饰,java8中
- 内部类访问局部变量的时候,为什么变量必须加上final修饰(转)
- 内部类访问局部变量的时候,为什么变量必须加上final修饰
- 为什么内部类访问局部变量需要加final修饰?
- 为什么方法中的内部类只能访问final修饰的局部变量
- Java基础-内部类-为什么局部和匿名内部类只能访问局部final变量
- 关于java为什么在Method中局部内部类或者匿名内部类调用Method中定义的变量要用final修饰?
- 为什么Java匿名内部类的方法中用到的局部变量都必须定义为final
- Java中为什么匿名内部类和局部内部类只能访问final变量?
- 为什么局部内部类和匿名内部类只能访问final的局部变量?
- Java--局部内部类在访问他所在方法中的局部变量必须用final修饰