談談final的作用
原文:https://www.cnblogs.com/xrq730/p/4820296.html
努力加油!
前言
一直想寫寫這個話題。代表公司也面試過一些求職者,每次面試我必問的兩個問題之一就是“請你談一談對於final關鍵字的理解”。這是一個簡單的小問題,但是不要小看它,通過對這個問題的回答以及一些簡單的擴展就能看出一個求職者的基礎是否紮實以及是否有鉆研精神。OK,開始進入主題。
final關鍵字的作用
相信對於final的用法,大多數人都可以隨口說出三句話:
1、被final修飾的類不可以被繼承
2、被final修飾的方法不可以被重寫
3、被final修飾的變量不可以被改變
重點就是第三句。被final修飾的變量不可以被改變,什麽不可以被改變呢,是變量的引用?還是變量裏面的內容?還是兩者都不可以被改變?寫個例子看一下就知道了:
1 public class FinalString 2 { 3 private String str; 4 5 public FinalString(String str) 6 { 7 this.str = str; 8 } 9 10 public String getStr() 11 { 12 return str; 13 } 14 15 public void setStr(String str) 16 { 17 this.str = str;18 } 19 }
1 public class Test 2 { 3 public static void main(String[] args) 4 { 5 final FinalString fs = new FinalString("1"); 6 fs.setStr("2"); 7 System.out.println(fs.getStr()); 8 } 9 }
運行一下,一點問題都沒有。稍微修改一下呢:
1 public static void main(String[] args) 2 {3 final FinalString fs = new FinalString("1"); 4 final FinalString fss = new FinalString("333"); 5 fs = fss; 6 }
第5行報錯了,“The final local variable fs cannot be assigned”。可見,被final修飾不可變的是變量的引用,而不是引用指向的內容,引用指向的內容是可以改變的。OK,那final修飾數組呢?
1 public static void main(String[] args) 2 { 3 final String[] strs0 = {"123","234"}; 4 final String[] strs1 = {"345","456"}; 5 strs1 = strs0; 6 strs1[1] = "333"; 7 }
同樣,第5行報錯了“The final local variable strs1 cannot be assigned”,第6行一點問題都沒有。變量和數組一樣,都是引用不可變,引用指向的內容可變。實際上如果用過FindBugs插件的應該知道,假如代碼裏面用final修飾了一個數組,那麽改行代碼會被作為findBugs的一個bug被查找出來,因為“用final修飾數組是沒有意義的”。
接下來,再看一下用final修飾方法參數的場景:
1 public class Test 2 { 3 public static void main(String[] args) 4 { 5 FinalString fs = new FinalString(""); 6 A(fs); 7 } 8 9 private static void A(final FinalString fs) 10 { 11 fs.setStr("123"); 12 FinalString fss = new FinalString("22"); 13 fs = fss; 14 } 15 }
一樣,同樣是13行報錯,11行沒有問題,相信大家已經知道原因了。
總結
“引用”是Java中非常重要的一個概念,對於引用的理解不深,很容易犯一些自己都沒有意識到的錯誤。被final修飾的變量,不管變量是在是哪種變量,切記不可變的是變量的引用而非引用指向對象的內容。另外,本文中關於final的作用還有兩點沒有講到:
1、被final修飾的方法,JVM會嘗試為之尋求內聯,這對於提升Java的效率是非常重要的。因此,假如能確定方法不會被繼承,那麽盡量將方法定義為final的,具體參見運行期優化技術的方法內聯部分
2、被final修飾的常量,在編譯階段會存入調用類的常量池中,具體參見類加載機制最後部分和Java內存區域
前言
一直想寫寫這個話題。代表公司也面試過一些求職者,每次面試我必問的兩個問題之一就是“請你談一談對於final關鍵字的理解”。這是一個簡單的小問題,但是不要小看它,通過對這個問題的回答以及一些簡單的擴展就能看出一個求職者的基礎是否紮實以及是否有鉆研精神。OK,開始進入主題。
final關鍵字的作用
相信對於final的用法,大多數人都可以隨口說出三句話:
1、被final修飾的類不可以被繼承
2、被final修飾的方法不可以被重寫
3、被final修飾的變量不可以被改變
重點就是第三句。被final修飾的變量不可以被改變,什麽不可以被改變呢,是變量的引用?還是變量裏面的內容?還是兩者都不可以被改變?寫個例子看一下就知道了:
1 public class FinalString 2 { 3 private String str; 4 5 public FinalString(String str) 6 { 7 this.str = str; 8 } 9 10 public String getStr() 11 { 12 return str; 13 } 14 15 public void setStr(String str) 16 { 17 this.str = str; 18 } 19 }按 Ctrl+C 復制代碼 按 Ctrl+C 復制代碼
運行一下,一點問題都沒有。稍微修改一下呢:
1 public static void main(String[] args) 2 { 3 final FinalString fs = new FinalString("1"); 4 final FinalString fss = new FinalString("333"); 5 fs = fss; 6 }
第5行報錯了,“The final local variable fs cannot be assigned”。可見,被final修飾不可變的是變量的引用,而不是引用指向的內容,引用指向的內容是可以改變的。OK,那final修飾數組呢?
按 Ctrl+C 復制代碼 按 Ctrl+C 復制代碼同樣,第5行報錯了“The final local variable strs1 cannot be assigned”,第6行一點問題都沒有。變量和數組一樣,都是引用不可變,引用指向的內容可變。實際上如果用過FindBugs插件的應該知道,假如代碼裏面用final修飾了一個數組,那麽改行代碼會被作為findBugs的一個bug被查找出來,因為“用final修飾數組是沒有意義的”。
接下來,再看一下用final修飾方法參數的場景:
1 public class Test 2 { 3 public static void main(String[] args) 4 { 5 FinalString fs = new FinalString(""); 6 A(fs); 7 } 8 9 private static void A(final FinalString fs) 10 { 11 fs.setStr("123"); 12 FinalString fss = new FinalString("22"); 13 fs = fss; 14 } 15 }
一樣,同樣是13行報錯,11行沒有問題,相信大家已經知道原因了。
總結
“引用”是Java中非常重要的一個概念,對於引用的理解不深,很容易犯一些自己都沒有意識到的錯誤。被final修飾的變量,不管變量是在是哪種變量,切記不可變的是變量的引用而非引用指向對象的內容。另外,本文中關於final的作用還有兩點沒有講到:
1、被final修飾的方法,JVM會嘗試為之尋求內聯,這對於提升Java的效率是非常重要的。因此,假如能確定方法不會被繼承,那麽盡量將方法定義為final的,具體參見運行期優化技術的方法內聯部分
2、被final修飾的常量,在編譯階段會存入調用類的常量池中,具體參見類加載機制最後部分和Java內存區域
談談final的作用