關(guān)于this逃逸

做了道選擇題,有個(gè)選項(xiàng)是這樣的

如果在一個(gè)線程構(gòu)造了一個(gè)不可變對(duì)象之后(對(duì)象僅包含final字段),就可以保證了這個(gè)對(duì)象被其他線程正確的查看

這個(gè)選項(xiàng)是錯(cuò)誤的

原因在這 http://ifeve.com/jmm-faq-finalright/

一個(gè)對(duì)象的final字段值是在它的構(gòu)造方法里面設(shè)置的。假設(shè)對(duì)象被正確的構(gòu)造了,一旦對(duì)象被構(gòu)造,在構(gòu)造方法里面設(shè)置給final字段的的值在沒有同步的情況下對(duì)所有其他的線程都會(huì)可見。另外,引用這些final字段的對(duì)象或數(shù)組都將會(huì)看到final字段的最新值。

對(duì)一個(gè)對(duì)象來說,被正確的構(gòu)造是什么意思呢?簡(jiǎn)單來說,它意味著這個(gè)正在構(gòu)造的對(duì)象的引用在構(gòu)造期間沒有被允許逸出。(參見安全構(gòu)造技術(shù))。換句話說,不要讓其他線程在其他地方能夠看見一個(gè)構(gòu)造期間的對(duì)象引用。不要指派給一個(gè)靜態(tài)字段,不要作為一個(gè)listener注冊(cè)給其他對(duì)象等等。這些操作應(yīng)該在構(gòu)造方法之后完成,而不是構(gòu)造方法中來完成。

class FinalFieldExample {
  final int x;
  int y;
  static FinalFieldExample f;
  public FinalFieldExample() {
    x = 3;
    y = 4;
  }

  static void writer() {
    f = new FinalFieldExample();
  }

  static void reader() {
    if (f != null) {
      int i = f.x;
      int j = f.y;
    }
  }
}```

上面的類展示了final字段應(yīng)該如何使用。一個(gè)正在執(zhí)行reader方法的線程保證看到f.x的值為3,因?yàn)樗莊inal字段。它不保證看到f.y的值為4,因?yàn)閒.y不是final字段。如果FinalFieldExample的構(gòu)造方法像這樣:

public FinalFieldExample() { // bad!
x = 3;
y = 4;
// bad construction - allowing this to escape
global.obj = this;
}


那么,從global.obj中讀取this的引用線程不會(huì)保證讀取到的x的值為3。

能夠看到字段的正確的構(gòu)造值固然不錯(cuò),但是,如果字段本身就是一個(gè)引用,那么,你還是希望你的代碼能夠看到引用所指向的這個(gè)對(duì)象(或者數(shù)組)的最新值。如果你的字段是final字段,那么這是能夠保證的。因此,當(dāng)一個(gè)final指針指向一個(gè)數(shù)組,你不需要擔(dān)心線程能夠看到引用的最新值卻看不到引用所指向的數(shù)組的最新值。重復(fù)一下,這兒的“正確的”的意思是“對(duì)象構(gòu)造方法結(jié)尾的最新的值”而不是“最新可用的值”。

現(xiàn)在,在講了如上的這段之后,如果在一個(gè)線程構(gòu)造了一個(gè)不可變對(duì)象之后(對(duì)象僅包含final字段),你希望保證這個(gè)對(duì)象被其他線程正確的查看,你仍然需要使用同步才行。例如,沒有其他的方式可以保證不可變對(duì)象的引用將被第二個(gè)線程看到。使用final字段的程序應(yīng)該仔細(xì)的調(diào)試,這需要深入而且仔細(xì)的理解并發(fā)在你的代碼中是如何被管理的。

如果你使用JNI來改變你的final字段,這方面的行為是沒有定義的。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容