匿名内部类访问的局部变量用final修饰

以前开发中一直会把匿名内部类访问的局部变量用final修饰。
最近开发中发现jdk8中其实可以去除final,编译器自动做了编译优化自动添加final修饰符。确实算是非常后知后觉了。。
首先回顾下为什么要添加final:
用final修饰实际上就是为了保护数据的一致性。
这里所说的数据一致性,对引用变量来说是引用地址的一致性,对基本类型来说就是值的一致性。

如果局部变量发生变化后,匿名内部类是不知道的(因为他只是拷贝了j局部变量的值,并不是直接使用的局部变量)。原先局部变量指向的是对象A,在创建匿名内部类后,匿名内部类中的成员变量也指向A对象。但过了一段时间局部变量的值指向另外一个B对象,但此时匿名内部类中还是指向原先的A对象。那么程序再接着运行下去,可能就会导致程序运行的结果与预期不同。

在JDK8中如果我们在匿名内部类中需要访问局部变量,那么这个局部变量不需要用final修饰符修饰。这是编译机制做的改变优化还是实际上就是一个语法糖(底层还是帮你加了final),通过下面的测试来进行验证(部分疑惑可能是由于反编译器的问题来引起的):
java代码:

public class TestInnerClass {
    private String hhh="hhh";

    public static void main(String[] args) {
        String str="hello";
        new Thread() {
            @Override
            public void run() {
                System.out.println(str);
            }
        }.start();
        new TestInnerClass().testFinal("wwww");
    }

    private void testFinal(String strK){
        new Thread() {
            @Override
            public void run() {
                System.out.println("testFinal: " + strK);
                System.out.println("testFinalhhh: " + hhh);
            }
        }.start();
    }
}

编译后jd-gui反编译后的代码-TestInnerClass,这里testFinal参数添加了final

import java.io.PrintStream;

public class TestInnerClass
{
  private String hhh = "hhh";
  
  public static void main(String[] args)
  {
    String str = "hello";
    new Thread()
    {
      public void run()
      {
        System.out.println(this.val$str);
      }
    }.start();
    new TestInnerClass().testFinal("wwww");
  }
  
  private void testFinal(final String strK)
  {
    new Thread()
    {
      public void run()
      {
        System.out.println("testFinal: " + strK);
        System.out.println("testFinalhhh: " + TestInnerClass.this.hhh);
      }
    }.start();
  }
}

编译后jd-gui反编译后的代码-TestInnerClass$1-只是拷贝了局部变量的值,并不是直接使用的局部变量

import java.io.PrintStream;

final class TestInnerClass$1
  extends Thread
{
  TestInnerClass$1(String paramString) 
  {
    //反编译工具的疏忽实际上隐藏了下面的代码
    //this.val$str = paramString;
  }
  
  public void run()
  {
    System.out.println(this.val$str);
  }
}

编译后jd-gui反编译后的代码-TestInnerClass$2-只是拷贝了j局部变量的值,并不是直接使用的局部变量

import java.io.PrintStream;

class TestInnerClass$2
  extends Thread
{
  TestInnerClass$2(TestInnerClass this$0, String paramString) 
  {
    //反编译工具的疏忽实际上隐藏了下面的代码
    //this.val$strK= paramString;
    //this.this$0= this$0;
  }
  
  public void run()
  {
    System.out.println("testFinal: " + this.val$strK);
    //TestInnerClass.access$000(this.this$0) 实际上就是 this.this$0.hhh
    System.out.println("testFinalhhh: " + TestInnerClass.access$000(this.this$0));
  }
}

参考博客:
内部类详解
内部类局部变量final修饰

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值