https://blog.csdn.net/z55887/article/details/49229491

先抛出让我疑惑了很久的一个问题

编程时,在线程中使用局部变量时候经常编译器会提示:局部变量必须声明为final

package test;

public class ThreadTest {

public void function(String a) {

new Thread(){
@Override
public void run() {
System.out.println(a);
}
}.start();
}

public static void main(String[] args) {
new ThreadTest().function("a");

}
}

上图中由于方法function中的形参a没有声明为final,编译抛出异常:Cannot refer to the non-final local variable a defined in an enclosing scope

这个问题我特意问过老师,也百度过,都没有给出满意的解答。今天看安卓视频无意发现了答案,真是意外之喜啊!

其实原因就是一个规则:java内部类访问局部变量时局部变量必须声明为final。

那为什么要这样呢?还有线程为什么和内部类一样?接下来我们慢慢揭秘。

public class Out {

public void test(final String a) {
class In{

public void function() {
System.out.println(a);
}
}
new In().function();
}

public static void main(String[] args) {
new Out().test("hi");
}
}
编译这个类后发现产生了两个class文件

也就是说内部类和外部类各一个class文件,这样就产生了一个问题,调用内部类方法的时候如何访问外部类方法中的局部变量呢?

实际上编译后的内部类的构造方法的里面,传了对应的外部类的引用和所有局部变量的形参。

(由于外部类方法执行完后局部变量会消亡,所以内部类构造函数中的局部变量实际是一份“复制”。而为了访问外部类中的私有成员变量,外部类编译后也产生了访问类似与getXXX的方法。)

这时产生了一个不一致的问题,如果局部变量不设为final,那内部类构造完毕后,外部类的局部变量又改变了那怎么办?

public class Out {

public void test(String a) {
class In{
public void function() {
System.out.println(a);
}
}
a="hello";
new In().function();
}

public static void main(String[] args) {
new Out().test("hi");
}
}

如代码中所示,这样调用内部类方法时会造成外部类局部变量和内部类中对应的变量的不一致。(注意内部类编译成class文件与new无关,a="hello"放在new In()前后不影响不一致关系,new在jvm运行class文件时才起效)

理解完内部类必须访问final声明的局部变量原因,我们回到最开始的问题:为什么线程和内部类一样

因为线程也是一个类,所以new Thread也相当于创建了一个内部类啦

我们编译一下最开始的ThreadTest.java文件

发现线程编译后也是产生了单独的class文件。

至此,问题全部解决啦~~

最后说明一下java1.8和之前版本对这个规则编译的区别。

如果在1.8的环境下,会很神奇的发现我们最开始的ThreadTest.java文件编译和运行是完全没有问题的,也就是说内部类使用的局部变量是可以不声明为final?!

且慢,如果我们给局部变量再赋下值会发现编译又会出现同样的错误

public class ThreadTest {

public void function(String a) {
a="b";
new Thread(){
@Override
public void run() {
System.out.println(a);
}
}.start();
}

public static void main(String[] args) {
new ThreadTest().function("a");

}
}
在a="b"这一行报错:Local variable a defined in an enclosing scope must be final or effectively final
也就是说规则没有改变,只是java1.8的编译变得更加智能了而已,在局部变量没有重新赋值的情况下,它默认局部变量为final型,认为你只是忘记加final声明了而已。如果你重新给局部变量改变了值或引用,那就无法默认为final了,所以报错。

参考网站:

详细原理版:java内部类访问局部变量时的final问题 https://blog.csdn.net/xiancaieeee/article/details/8834352

对照原理版:关于java里方法的内部类只能访问被final修饰的局部变量和... http://bbs.itheima.com/thread-136974-1-1.html

jdk不同带来的区别:Java中方法内定义的内部类可以访问方法中的局部变量的问题 https://bbs.csdn.net/topics/390918289?page=1

最新文章

  1. MVC5 + EF6 简单示例
  2. MATLAB寻找数组前k个大值
  3. TextBox的值是否为数字
  4. winrt反射
  5. HDU 5353
  6. (C/C++) Callback Function 回调(diao)函数
  7. Mysql 复制工具(percona-toolkit)
  8. CodeForces 173C Spiral Maximum (想法、模拟)
  9. 【转载】总结一下Android中主题(Theme)的正确玩法
  10. 使用Userlock监控用户访问 增强学校网络安全
  11. Struts1项目转成Struts2项目步奏
  12. BZOJ:4659&&BZOJ:2694: Lcm
  13. sql 选取分组中的第一条,显示聚合以外的列,having字句的使用
  14. 『NOIP2018普及组题解』
  15. Java开发笔记(五十六)利用枚举类型实现高级常量
  16. Omi框架学习之旅 - 通过对象实例来实现组件通讯 及原理说明
  17. ettercap中间人攻击--参数介绍
  18. Error: Cannot find module PhantomJS
  19. Http 服务 简单示例
  20. php实现头像预览上传功能

热门文章

  1. mysql首次使用过程以及彻底卸载过程
  2. GBDT的理解和总结
  3. css3 transform 变形属性详解
  4. Chrome开发者调试工具
  5. MySQL 之全文索引
  6. python运算符和常用数据类型转换
  7. eclipse、myeclipse使用常用的小技巧
  8. Java中的字符串比较
  9. es6 转载
  10. 关于运算符的那些坑—自增x++&&++y