我們給匿名內(nèi)部類傳遞參數(shù)的時(shí)候贰您,若該形參在內(nèi)部類中需要被使用,那么該形參必須要為final拢操。也就是說:**當(dāng)所在的方法的形參需要被內(nèi)部類里面使用時(shí)锦亦,該形參必須為final。
為什么必須要為final呢令境?
首先我們知道在內(nèi)部類編譯成功后杠园,它會(huì)產(chǎn)生一個(gè)class文件,該class文件與外部類并不是同一class文件舔庶,僅僅只保留對(duì)外部類的引用抛蚁。當(dāng)外部類傳入的參數(shù)需要被內(nèi)部類調(diào)用時(shí),從java程序的角度來看是直接被調(diào)用:
public class OuterClass {
public void display(final String name,String age){
class InnerClass{
void display(){
System.out.println(name);
}
}
}
}
從上面代碼中看好像name參數(shù)應(yīng)該是被內(nèi)部類直接調(diào)用惕橙?其實(shí)不然篮绿,在java編譯之后實(shí)際的操作如下:
public class OuterClassname = name;
this.InnerClass$age = age;
}
public void display(){
System.out.println(this.InnerClass$name + "----" + this.InnerClass$age );
}
}
所以從上面代碼來看,內(nèi)部類并不是直接調(diào)用方法傳遞的參數(shù)吕漂,而是利用自身的構(gòu)造器對(duì)傳入的參數(shù)進(jìn)行備份,自己內(nèi)部方法調(diào)用的實(shí)際上時(shí)自己的屬性而不是外部方法傳遞進(jìn)來的參數(shù)尘应。
直到這里還沒有解釋為什么是final惶凝?在內(nèi)部類中的屬性和外部方法的參數(shù)兩者從外表上看是同一個(gè)東西,但實(shí)際上卻不是犬钢,所以他們兩者是可以任意變化的苍鲜,也就是說在內(nèi)部類中我對(duì)屬性的改變并不會(huì)影響到外部的形參,而然這從程序員的角度來看這是不可行的玷犹,畢竟站在程序的角度來看這兩個(gè)根本就是同一個(gè)混滔,如果內(nèi)部類該變了,而外部方法的形參卻沒有改變這是難以理解和不可接受的,所以為了保持參數(shù)的一致性坯屿,就規(guī)定使用final來避免形參的不改變油湖。
簡(jiǎn)單理解就是,拷貝引用领跛,為了避免引用值發(fā)生改變乏德,例如被外部類的方法修改等,而導(dǎo)致內(nèi)部類得到的值不一致吠昭,于是用final來讓該引用不可改變喊括。
故如果定義了一個(gè)匿名內(nèi)部類,并且希望它使用一個(gè)其外部定義的參數(shù)矢棚,那么編譯器會(huì)要求該參數(shù)引用是final的郑什。
同理,內(nèi)部類導(dǎo)致外部類泄漏也可理解了蒲肋,非靜態(tài)內(nèi)部類持有外部類的引用
1蘑拯、編譯器自動(dòng)為內(nèi)部類添加一個(gè)成員變量, 這個(gè)成員變量的類型和外部類的類型相同肉津, 這個(gè)成員變量就是指向外部類對(duì)象(this)的引用强胰;
2、編譯器自動(dòng)為內(nèi)部類的構(gòu)造方法添加一個(gè)參數(shù)妹沙, 參數(shù)的類型是外部類的類型偶洋, 在構(gòu)造方法內(nèi)部使用這個(gè)參數(shù)為內(nèi)部類中添加的成員變量賦值;
3距糖、在調(diào)用內(nèi)部類的構(gòu)造函數(shù)初始化內(nèi)部類對(duì)象時(shí)玄窝,會(huì)默認(rèn)傳入外部類的引用。