1. 程式人生 > >java-final關鍵字修飾變數

java-final關鍵字修飾變數

       在java中,final關鍵字可以對類、方法以及變數進行修飾,這裡對final修飾變數進行一些探討。

       final修飾的變數不可更改,其不可更改指的是其引用不可修改。看下列程式碼:

StringBuffer sb1 = new StringBuffer("sb");
StringBuffer sb2 = sb1;
final StringBuffer finalSb = sb2;
System.out.println(finalSb);//輸出: sb
		
sb1.append(" sb1 append");
System.out.println(finalSb);//輸出: sb sb1 append
		
sb2.append(" sb2 append");
System.out.println(finalSb);//輸出: sb sb1 append sb2 append

        在程式碼中,無論是對sb1還是sb2進行操作,均可以影響到finalSb的字面值,看似是finalSb → sb2 → sb1,這樣一個變數指向另外一個變數,將變數的值複製傳遞下去。

       這裡進行一個猜想,原來是finalSb → sb2 → sb1,現在將sb2 → new StringBuffer(" new sb2"),finalSb的字面值是否變化呢?我們看程式碼輸出:

StringBuffer sb1 = new StringBuffer("sb");
StringBuffer sb2 = sb1;
final StringBuffer finalSb = sb2;
System.out.println(finalSb);//輸出: sb
		
sb1.append(" sb1 append");
System.out.println(finalSb);//輸出: sb sb1 append
		
sb2.append(" sb2 append");
System.out.println(finalSb);//輸出: sb sb1 append sb2 append
		
sb2 = new StringBuffer("new sb2");
System.out.println(finalSb);//輸出: sb sb1 append sb2 append

sb2.append(" xx app");//對sb2進行操作
System.out.println(finalSb);//輸出: sb sb1 append sb2 append

       通過程式碼的輸出可以看到,哪怕是將sb2重新指向了另外一個新的引用,finalSb的字面值並不會發生改變,對sb2進行的操作,也自然不會影響到finalSb的字面值。

       sb1和原來的sb2操作時,可以影響到finalSb的值,新的sb2又不可以影響到finalSb的值,這是為什麼呢?這裡我畫出示意圖如下:

        

       sb2並不是直接指向sb1,而是指向的是sb1在堆空間中new出來的物件,同樣的,finalSb也不是指向的sb2,而是指向的sb2所指向的堆空間的物件,因為三者均是指向的同一個物件,所以,無論是對哪一個變數進行操作,均會影響到其字面值。

       當執行 sb2 = new StringBuffer("new sb2") 時,sb2不再執行堆空間原來的物件,而是指向了新的物件,如圖:

     

       sb2已經指向了新的地址,故而對sb2進行怎麼樣的操作,finalSb的字面值均不會發生變化。

       從以上兩張圖和程式碼可以看出,final所修飾的變數,其引用不可變(或者是其在記憶體中指向的地址不可更改),還有就是,基於引用本身的操作(如List型別的add,StringBuffer的append)不受影響

       另外,final修飾的變數還被賦予了編譯常量的功能,如程式碼所示:

final String finalStrA = "str";
String strA = "str";
String strB = "str2";
String strC = finalStrA+"2";
String strD = strA+"2";
String strE = "str"+"2";//等同於 finalStrA+"2"
		
System.out.println(strB==strC);//輸出: true
System.out.println(strB==strD);//輸出: false
System.out.println(strB==strE);//輸出: true