[Java] 在匿名内部类中使用参数需要添加final 的解释

今天在看Thinking in Java时候 遇上的小问题 做一个小整理

如果我在外部类中定义一个匿名内部类的时候 需要在使用在其外部的参数的时候 需要在调入参数的时候 给参数加 final。如果不加的话 会得到一个编译错误。

package learnInnerClass;

interface Inner{
    void f();
}
public class Outter {
    public Inner destination(final int x){
        return new Inner(){
            public void f(){
                System.out.println(x);
            }
        };
    }
    public static void main(String[] argv){
        Outter outter = new Outter();
        Inner inner = outter.destination(5);
        inner.f();
    }
}
/*
5
*/

但是呢 如果我不使用 就不用添加了final了

package learnInnerClass;

interface Inner{
    void f();
}
public class Outter {
    public Inner destination(int x){
        return new Inner(){
            public void f(){
                System.out.println("I don't use x");
            }
        };
    }
    public static void main(String[] argv){
        Outter outter = new Outter();
        Inner inner = outter.destination(5);
        inner.f();
    }
}
/*
I don't use x
*/

那为什么 我们一定要添加final标记呢 如果用一句话来说 就是保持内部类和外部的数据一致性

这里参考了jlustone老哥的解释 从编译器的角度解释了这个问题(https://www.cnblogs.com/jlustone/p/7517323.html

首先,内部类被编译的时候会生成一个单独的内部类的.class文件,这个文件并不与外部类在同一class文件中。
当外部类传的参数被内部类调用时,从java程序的角度来看是直接的调用例如:

public void dosome(final String a,final int b){ 
  class Dosome{public void dosome(){System.out.println(a+b)}}; 
  Dosome some=new Dosome(); 
  some.dosome(); 
}

从代码来看好像是那个内部类直接调用的a参数和b参数,但是实际上不是,在java编译器编译以后实际的操作代码是

class Outer$Dosome{ 
    public Dosome(final String a,final int b){ 
    this.Dosome$a=a; 
    this.Dosome$b=b; 
    } 
     public void dosome(){ 
       System.out.println(this.Dosome$a+this.Dosome$b); 
    } 
}

从以上代码看来,内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到了自己的内部,自己内部的方法调用的实际是自己的属性而不是外部类方法的参数。
这样理解就很容易得出为什么要用final了,因为两者从外表看起来是同一个东西,实际上却不是这样,如果内部类改掉了这些参数的值也不可能影响到原参数,然而这样却失去了参数的一致性,因为从编程人员的角度来看他们是同一个东西,如果编程人员在程序设计的时候在内部类中改掉参数的值,但是外部调用的时候又发现值其实没有被改掉,这就让人非常的难以理解和接受,为了避免这种尴尬的问题存在,所以编译器设计人员把内部类能够使用的参数设定为必须是final来规避这种莫名其妙错误的存在。

(简单理解就是,拷贝引用,为了避免引用值发生改变,例如被外部类的方法修改等,而导致内部类得到的值不一致,于是用final来让该引用不可改变)


不过Java8取消了这个设定… …

package learnInnerClass;

interface Inner{
    void f();
}
public class Outter {
    public Inner destination(int x){
        return new Inner(){
            public void f(){
                System.out.println(x);
            }
        };
    }
    public static void main(String[] argv){
        Outter outter = new Outter();
        Inner inner = outter.destination(5);
        inner.f();
    }
}
/*
5
*/
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值