匿名内部类局部变量final问题

匿名内部类局部变量为什么要加final?

先来看一段代码:

/**
 * Author: wang bo
 * Date:2019/1/30
 * Description:
 */
public class FinalTest {

    private ArrayList list = new ArrayList();

    public void f1() {

        final Test test = new Test();
        list.add(new Listener() {
            @Override
            public Test listen() {
                Test t = test;
                return t;
            }
        });
    }
    private interface Listener {
        Test listen();
    }

    class Test {
    }
}
复制代码

可以看到想在匿名内部类里面return t,外面的test必须加上final修饰符。这是为什么呢?这是java的语言基础,网上这个问题也被讨论烂了,各种说法都有,所以一开始我被弄的稀里糊涂,请教了公司前辈后,再加上自己的理解,算是弄懂了吧。

抛出疑问

1.如果f1()函数结束出栈,test会被回收吗?

2.如果被回收,那么test引用指向的对象会被gc掉吗?


我们先来看看这段代码用JD-GUI反编译后的结果:

一共有4个文件:

FinalTest$1.class 这是虚拟机为匿名内部类创建的实现类

FinalTest$Listener.class 这是FinalTest的内部接口

FinalTest$Test.class 这是内部类Test

FinalTest.class 这是FinalTest


要理解这个问题重要的是FinalTest$1.class, 和 FinalTest.class 俩个文件。

先抛出答案:

1.如果f1()函数结束出栈,test引用会被回收吗?

2.如果被回收,那么test引用指向的对象会被gc掉吗?不会

FinalTest.class


public class FinalTest
{
  private ArrayList list = new ArrayList();
  
  public void f1()
  {
    final Test test = new Test();
    this.list.add(new Listener()
    {
      public FinalTest.Test listen()
      {
        FinalTest.Test t = test;
        return t;
      }
    });
  }
  
  class Test
  {
    Test() {}
  }
  
  private static abstract interface Listener
  {
    public abstract FinalTest.Test listen();
  }
}
复制代码

可以看到反编译后test 还是被final修饰的。再来看看FinalTest$1.class。


class FinalTest$1
  implements FinalTest.Listener
{
  FinalTest$1(FinalTest this$0, FinalTest.Test paramTest) {}
  
  public FinalTest.Test listen()
  {
    FinalTest.Test t = this.val$test;
    return t;
  }
}
复制代码

这个反编译后的结果就是关键,可以看到匿名内部类的实现类的构造函数的参数是 this$0paramTest ,这个this$0 是外部类的引用,而paramTest 是需要接收刚刚传入的test局部变量的,并且我们知道java是值传递的,所以test局部变量的引用的值被“拷贝了一份”指向了同一个地址。

继续看listen方法,我们可以看到this.val$test ,不知道是不是反编译工具的问题,在构造函数中应该是有

this.val$test = paramTest 这样的语句在的。

所以这种现象可以解决第一问和第二问了,函数退出test引用被销毁。而在匿名内部类的实现类里,拷贝的引用生命周期和匿名内部类实现类对应的对象的生命周期一样长。而且因为对应的对象存在强引用(拷贝的引用),所以不会被gc。

所以为了解决数据一致性的问题,java要保证匿名内部类的里面拷贝的引用和局部变量的引用指向的是同一个对象,所以局部变量要加final,保证了拷贝的引用和它指向的是同一个对象。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
局部内部类和匿名内部类Java中的两种特殊类型的内部类。 局部内部类是定义在方法或者代码块内部的类。它的作用域被限定在所属的方法或代码块内部,不能在外部访问。局部内部类可以访问所属方法或代码块的局部变量,但是这些变量必须声明为final或者事实上的final(在Java 8之后,可以省略final关键字,但是不能对这些变量进行修改)。局部内部类的定义形式如下: ```java public class OuterClass { public void someMethod() { final int localVar = 10; class LocalInnerClass { public void innerMethod() { System.out.println(localVar); } } LocalInnerClass inner = new LocalInnerClass(); inner.innerMethod(); } } ``` 匿名内部类是没有名字的内部类,它直接继承了一个父类或者实现了一个接口。通常用于创建只使用一次的简单类。匿名内部类没有构造方法,因此可以直接通过实例化后的对象进行方法调用。匿名内部类的定义形式如下: ```java public class OuterClass { public void someMethod() { Thread thread = new Thread(new Runnable() { public void run() { System.out.println("Thread is running"); } }); thread.start(); } } ``` 在上述代码中,我们使用匿名内部类实现了Runnable接口,并在Thread类的构造方法中传入了该匿名内部类的实例。 总结一下,局部内部类是定义在方法或代码块内部的类,它的作用域被限定在所属的方法或代码块内部;而匿名内部类是没有名字的内部类,通常用于创建只使用一次的简单类。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值