再次認識下final關鍵字和不變性

什麼是不變性

  • 如果對象被創建後,狀態就不能被修改了,那麼它就是不可變的
  • 如:person對象的birthday和sex被設置成final的,那麼一旦創建了就不可變的
public class Persion {
    private final Date birthday = new Date();
    private final int sex = 0;
}
  • 具有不變性的對象一定是線程安全的

final的作用

  • 類防止被繼承
  • 方法防止被重寫
  • 變量防止被篡改
  • 爲了保證線程安全且不使用額外同步開銷

final的三種用法

final修飾變量

  • 含義:被final修飾的變量,意味着不能被修改,如果變量是對象,那麼對象的引用不能變,但是對象自身的內容依然可以變化

雖然person被設置爲final的變量,但是person對象的值依然是可以修改的

public static void main(String[] args) {
        final Persion persion = new Persion();
        persion.setName("mary");
       
}

賦值時機

  • 類中的final屬性
    • 聲明變量時直接在等號右邊賦值private final int a = 123;
    • 構造函數中賦值
    • 類的初始化代碼塊中賦值
			//構造函數中賦值
			private final String name ;
			public Persion(String name) {
			    this.name = name;
			}
---------------------------------------------------------------------
			//類的初始化代碼塊中賦值
			private final String name ;
		    {
		        name = "tom";
		    }
  • 類中的static final屬性
    • 聲明變量時直接在等號右邊賦值private static final int a = 123;
    • static代碼塊賦值
			private static final String name ;
		    static {
		        name = "tom";
		    }
  • 方法中的final變量
    • 不要求賦值時機,但是使用前必須賦值,和非final變量一致

final修飾方法

  • 構造方法不允許final修飾
  • 修飾的方法不可被重寫,即使子類有同樣名稱的方法,也不是重寫(和靜態方法一致)

final修飾類

  • 不可被繼承
  • 典型案例:String類

注意點

  • final修飾對象,只是對象的引用不可變,對象的屬性是可以變化的
  • 明確知道一個類創建後不會被變化,最好加一個final,提高代碼可讀性

不變性與final的關係

並不是意味着簡單的用final修飾就是不變性

  • 對於基本數據類型,被final修飾後就具有不可變性
  • 對於對象類型
    • 需要保證對象自身被創建後,狀態永遠不會變。
    • 所有屬性都是final修飾的
    • 對象創建過程沒有發生溢出

棧封閉技術

在方法裏新建的局部變量,實際上是存儲在每個線程的私有棧空間,而每個棧的棧空間是不會被其他線程訪問到的,所以不會有線程安全問題

實現一個runnable接口,在run方法中累加10000次,再調用一個擁有局部變量的方法,方法內同樣實現累加10000次
創建兩個線程使用上面創建的實現類

public class StackConfinement implements Runnable{

    int index =  0;
    public static void main(String[] args) throws InterruptedException {
        StackConfinement r1 = new StackConfinement();
        Thread thread1 = new Thread(r1);
        Thread thread2 = new Thread(r1);
        thread1.start();
        thread2.start();
        thread1.join();
        thread2.join();
        System.out.println(r1.index);
    }

    public void inThread(){
        int neverGoOut = 0;
        for (int i = 0; i<10000;i++){
            neverGoOut++;
        }
        System.out.println("棧內保護的數據是線程安全的:"+neverGoOut);
    }

    @Override
    public void run() {
        for (int i = 0; i<10000;i++){
            index++;
        }
        inThread();
    }
}

打印結果如圖

棧內保護的數據是線程安全的:10000
棧內保護的數據是線程安全的:10000
15033

受棧空間保護的數據是線程安全的,而沒有被保護的數據是存在線程安全的(15033<20000)

面試題

推測下面一段代碼的運行結果:

  public static void main(String[] args) {
        String a = "test2";
        final String b = "test";
        String d = "test";
        String c = b + 2;
        String e = d + 2;
        System.out.println(a == c);
        System.out.println(a == e);
    }

運行結果爲:

true
false

分析:

  • 對於c: 其中b是被final修飾的,所以在編譯期間就知道b的準確值了,所以c ="test"+2,而編譯器會把"test"+2自動優化成"test2",將"test2"賦值給c時,首先會查詢常量池是否存在"test2",因爲對a賦值的時候已經在常量池創建了"test2",所以就直接將"test2"的引用指向c,所以a == c
  • 對於e: 其中d指向常量池中的"test",在編譯期並不知道其具體的值,所以需要在運行時才知道具體的值,對於運行期才知道值的情況,JVM會調用new String(e),e的值將在上被創建,而a在常量池,所以a != e

如果沒怎麼看懂,關於String在JVM中的存儲和編譯器優化可以參考我的博客《String是如何實現的?有哪些重要方法?》


本文參考了:《玩轉Java併發工具》


更多Java面試複習筆記和總結可訪問我的面試複習專欄《Java面試複習筆記》,或者訪問我另一篇博客《Java面試核心知識點彙總》查看目錄和直達鏈接

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章