Java 中的雙重檢查(Double-Check)

來源:互聯網
上載者:User

在 Effecitve Java 一書的第 48 條中提到了雙重檢查模式,並指出這種模式在 Java 中通常並不適用。該模式的結構如下所示:

 

public Resource getResource() {<br /> if (resource == null) {<br /> synchronized(this){<br /> if (resource==null) {<br /> resource = new Resource();<br /> }<br /> }<br /> }<br /> return resource;<br />}

 

該模式是對下面的代碼改進:

public synchronized Resource getResource(){<br /> if (resource == null){<br /> resource = new Resource();<br /> }<br /> return resource;<br />}

 

這段代碼的目的是對 resource 延遲初始化。但是每次訪問的時候都需要同步。為了減少同步的開銷,於是有了雙重檢查模式。

在 Java 中雙重檢查模式無效的原因是在不同步的情況下參考型別不是安全執行緒的。對於除了 long 和 double 的基本類型,雙重檢查模式是適用
的。比如下面這段代碼就是正確的:

private int count;<br />public int getCount(){<br /> if (count == 0){<br /> synchronized(this){<br /> if (count == 0){<br /> count = computeCount(); //一個耗時的計算<br /> }<br /> }<br /> }<br /> return count;<br />}

 

上面就是關於java中雙重檢查模式(double-check
idiom)的一般結論。但是事情還沒有結束,因為java的記憶體模式也在改進中。Doug
Lea 在他的文章中寫道:“根據最新的 JSR133 的 Java 記憶體模型,如果將參考型別聲明為 volatile,雙重檢查模式就可以工作了”,參見
http://gee.cs.oswego.edu/dl/cpj/updates.html

所以以後要在 Java 中使用雙重檢查模式,可以使用下面的代碼:

 

private volatile Resource resource;<br />public Resource getResource(){<br /> if (resource == null){<br /> synchronized(this){<br /> if (resource==null){<br /> resource = new Resource();<br /> }<br /> }<br /> }<br /> return resource;<br />}

 

當然了,得是在遵循 JSR133 規範的 Java 中。

所以,double-check 在 J2SE 1.4 或早期版本在多線程或者 JVM 調優時由於 out-of-order writes,是停用。
這個問題在 J2SE 5.0 中已經被修複,可以使用 volatile 關鍵字來保證多線程下的單例。

public class Singleton {<br /> private volatile Singleton instance = null;<br /> public Singleton getInstance() {<br /> if (instance == null) {<br /> synchronized(this) {<br /> if (instance == null) {<br /> instance = new Singleton();<br /> }<br /> }<br /> }<br /> return instance;<br /> }<br />}<br />

 

推薦方法
是Initialization on Demand Holder(IODH),

詳見 http://en.wikipedia.org/wiki/Initialization_on_demand_holder_idiom

 

public class Singleton {<br /> static class SingletonHolder {<br /> static Singleton instance = new Singleton();<br /> }</p><p> public static Singleton getInstance(){<br /> return SingletonHolder.instance;<br /> }<br />}<br />

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.