成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

雙重檢查鎖定失效分析

keke / 3344人閱讀

摘要:雙重檢查鎖定以下稱為已被廣泛當(dāng)做多線程環(huán)境下延遲初始化的一種高效手段。由于沒有對(duì)這些做出明確規(guī)定,很難說是否有效。可以在中使用顯式的內(nèi)存屏障來使生效,但中并沒有這些屏障。如果改變鎖釋放的語(yǔ)義釋放時(shí)執(zhí)行一個(gè)雙向的內(nèi)存屏障將會(huì)帶來性能損失。

雙重檢查鎖定(以下稱為DCL)已被廣泛當(dāng)做多線程環(huán)境下延遲初始化的一種高效手段。

遺憾的是,在Java中,如果沒有額外的同步,它并不可靠。在其它語(yǔ)言中,如c++,實(shí)現(xiàn)DCL,需要依賴于處理器的內(nèi)存模型、編譯器實(shí)行的重排序以及編譯器與同步庫(kù)之間的交互。由于c++沒有對(duì)這些做出明確規(guī)定,很難說DCL是否有效。可以在c++中使用顯式的內(nèi)存屏障來使DCL生效,但Java中并沒有這些屏障。

// Single threaded version
class Foo { 
  private Helper helper = null;
  public Helper getHelper() {
    if (helper == null) 
        helper = new Helper();
    return helper;
    }
  // other functions and members...
}

如果這段代碼用在多線程環(huán)境下,有幾個(gè)可能出錯(cuò)的地方。最明顯的是,可能會(huì)創(chuàng)建出兩或多個(gè)Helper對(duì)象。(后面會(huì)提到其它問題)。將getHelper()方法改為同步即可修復(fù)此問題。

// Correct multithreaded version
class Foo { 
  private Helper helper = null;
  public synchronized Helper getHelper() {
    if (helper == null) 
        helper = new Helper();
    return helper;
    }
  // other functions and members...
}

上面的代碼在每次調(diào)用getHelper時(shí)都會(huì)執(zhí)行同步操作。DCL模式旨在消除helper對(duì)象被創(chuàng)建后還需要的同步。

// Broken multithreaded version
// "Double-Checked Locking" idiom
class Foo { 
  private Helper helper = null;
  public Helper getHelper() {
    if (helper == null) 
      synchronized(this) {
        if (helper == null) 
          helper = new Helper();
      }    
    return helper;
    }
  // other functions and members...
}

不幸的是,這段代碼無論是在優(yōu)化型的編譯器下還是在共享內(nèi)存處理器中都不能有效工作。

不起作用

上面代碼不起作用的原因有很多。接下來我們先說幾個(gè)比較顯而易見的原因。理解這些之后,也許你想找出一種方法來“修復(fù)”DCL模式。你的修復(fù)也不會(huì)起作用:這里面有很微妙的原因。在理解了這些原因之后,可能想進(jìn)一步進(jìn)行修復(fù),但仍不會(huì)正常工作,因?yàn)榇嬖诟⒚畹脑颉?/p>

很多聰明的人在這上面花費(fèi)了很多時(shí)間。除了在每個(gè)線程訪問helper對(duì)象時(shí)執(zhí)行鎖操作別無他法。

不起作用的第一個(gè)原因

最顯而易見的原因是,Helper對(duì)象初始化時(shí)的寫操作與寫入helper字段的操作可以是無序的。這樣的話,如果某個(gè)線程調(diào)用getHelper()可能看到helper字段指向了一個(gè)Helper對(duì)象,但看到該對(duì)象里的字段值卻是默認(rèn)值,而不是在Helper構(gòu)造方法里設(shè)置的那些值。

如果編譯器將調(diào)用內(nèi)聯(lián)到構(gòu)造方法中,那么,如果編譯器能證明構(gòu)造方法不會(huì)拋出異?;驁?zhí)行同步操作,初始化對(duì)象的這些寫操作與hepler字段的寫操作之間就能自由的重排序。

即便編譯器不對(duì)這些寫操作重排序,在多處理器上,某個(gè)處理器或內(nèi)存系統(tǒng)也可能重排序這些寫操作,運(yùn)行在其它
處理器上的線程就可能看到重排序帶來的結(jié)果。

Doug Lea寫了一篇更詳細(xì)的有關(guān)編譯器重排序的文章。

展示其不起作用的測(cè)試案例

Paul Jakubik找到了一個(gè)使用DCL不能正常工作的例子。下面的代碼做了些許整理:

public class DoubleCheckTest
{


  // static data to aid in creating N singletons
  static final Object dummyObject = new Object(); // for reference init
  static final int A_VALUE = 256; // value to initialize "a" to
  static final int B_VALUE = 512; // value to initialize "b" to
  static final int C_VALUE = 1024;
  static ObjectHolder[] singletons;  // array of static references
  static Thread[] threads; // array of racing threads
  static int threadCount; // number of threads to create
  static int singletonCount; // number of singletons to create


  static volatile int recentSingleton;


  // I am going to set a couple of threads racing,
  // trying to create N singletons. Basically the
  // race is to initialize a single array of 
  // singleton references. The threads will use
  // double checked locking to control who 
  // initializes what. Any thread that does not
  // initialize a particular singleton will check 
  // to see if it sees a partially initialized view.
  // To keep from getting accidental synchronization,
  // each singleton is stored in an ObjectHolder 
  // and the ObjectHolder is used for 
  // synchronization. In the end the structure
  // is not exactly a singleton, but should be a
  // close enough approximation.
  // 


  // This class contains data and simulates a 
  // singleton. The static reference is stored in
  // a static array in DoubleCheckFail.
  static class Singleton
    {
    public int a;
    public int b;
    public int c;
    public Object dummy;

    public Singleton()
      {
      a = A_VALUE;
      b = B_VALUE;
      c = C_VALUE;
      dummy = dummyObject;
      }
    }

  static void checkSingleton(Singleton s, int index)
    {
    int s_a = s.a;
    int s_b = s.b;
    int s_c = s.c;
    Object s_d = s.dummy;
    if(s_a != A_VALUE)
      System.out.println("[" + index + "] Singleton.a not initialized " +
s_a);
    if(s_b != B_VALUE)
      System.out.println("[" + index 
                         + "] Singleton.b not intialized " + s_b);

    if(s_c != C_VALUE)
      System.out.println("[" + index 
                         + "] Singleton.c not intialized " + s_c);

    if(s_d != dummyObject)
      if(s_d == null)
        System.out.println("[" + index 
                           + "] Singleton.dummy not initialized," 
                           + " value is null");
      else
        System.out.println("[" + index 
                           + "] Singleton.dummy not initialized," 
                           + " value is garbage");
    }

  // Holder used for synchronization of 
  // singleton initialization. 
  static class ObjectHolder
    {
    public Singleton reference;
    }

  static class TestThread implements Runnable
    {
    public void run()
      {
      for(int i = 0; i < singletonCount; ++i)
        {
    ObjectHolder o = singletons[i];
        if(o.reference == null)
          {
          synchronized(o)
            {
            if (o.reference == null) {
              o.reference = new Singleton();
          recentSingleton = i;
          }
            // shouldn"t have to check singelton here
            // mutex should provide consistent view
            }
          }
        else {
          checkSingleton(o.reference, i);
      int j = recentSingleton-1;
      if (j > i) i = j;
      }
        } 
      }
    }

  public static void main(String[] args)
    {
    if( args.length != 2 )
      {
      System.err.println("usage: java DoubleCheckFail" +
                         "  ");
      }
    // read values from args
    threadCount = Integer.parseInt(args[0]);
    singletonCount = Integer.parseInt(args[1]);

    // create arrays
    threads = new Thread[threadCount];
    singletons = new ObjectHolder[singletonCount];

    // fill singleton array
    for(int i = 0; i < singletonCount; ++i)
      singletons[i] = new ObjectHolder();

    // fill thread array
    for(int i = 0; i < threadCount; ++i)
      threads[i] = new Thread( new TestThread() );

    // start threads
    for(int i = 0; i < threadCount; ++i)
      threads[i].start();

    // wait for threads to finish
    for(int i = 0; i < threadCount; ++i)
      {
      try
        {
        System.out.println("waiting to join " + i);
        threads[i].join();
        }
      catch(InterruptedException ex)
        {
        System.out.println("interrupted");
        }
      }
    System.out.println("done");
    }
}

當(dāng)上述代碼運(yùn)行在使用Symantec JIT的系統(tǒng)上時(shí),不能正常工作。尤其是,Symantec
JIT將

singletons[i].reference = new Singleton();

編譯成了下面這個(gè)樣子(Symantec JIT用了一種基于句柄的對(duì)象分配系統(tǒng))。

0206106A   mov         eax,0F97E78h
0206106F   call        01F6B210                  ; allocate space for
                                                 ; Singleton, return result in eax
02061074   mov         dword ptr [ebp],eax       ; EBP is &singletons[i].reference 
                                                ; store the unconstructed object here.
02061077   mov         ecx,dword ptr [eax]       ; dereference the handle to
                                                 ; get the raw pointer
02061079   mov         dword ptr [ecx],100h      ; Next 4 lines are
0206107F   mov         dword ptr [ecx+4],200h    ; Singleton"s inlined constructor
02061086   mov         dword ptr [ecx+8],400h
0206108D   mov         dword ptr [ecx+0Ch],0F84030h

如你所見,賦值給singletons[i].reference的操作在Singleton構(gòu)造方法之前做掉了。在現(xiàn)有的Java內(nèi)存模型下這完全是允許的,在c和c++中也是合法的(因?yàn)閏/c++都沒有內(nèi)存模型(譯者注:這篇文章寫作時(shí)間較久,c++11已經(jīng)有內(nèi)存模型了))。

一種不起作用的“修復(fù)”

基于前文解釋的原因,一些人提出了下面的代碼:

// (Still) Broken multithreaded version
// "Double-Checked Locking" idiom
class Foo { 
  private Helper helper = null;
  public Helper getHelper() {
    if (helper == null) {
      Helper h;
      synchronized(this) {
        h = helper;
        if (h == null) 
            synchronized (this) {
              h = new Helper();
            } // release inner synchronization lock
        helper = h;
        } 
      }    
    return helper;
    }
  // other functions and members...
}

將創(chuàng)建Helper對(duì)象的代碼放到了一個(gè)內(nèi)部的同步塊中。直覺的想法是,在退出同步塊的時(shí)候應(yīng)該有一個(gè)內(nèi)存屏障,這會(huì)阻止Helper的初始化與helper字段賦值之間的重排序。

很不幸,這種直覺完全錯(cuò)了。同步的規(guī)則不是這樣的。monitorexit(即,退出同步塊)的規(guī)則是,在monitorexit前面的action必須在該monitor釋放之前執(zhí)行。但是,并沒有哪里有規(guī)定說monitorexit后面的action不可以在monitor釋放之前執(zhí)行。因此,編譯器將賦值操作helper = h;挪到同步塊里面是非常合情合理的,這就回到了我們之前說到的問題上。許多處理器提供了這種單向的內(nèi)存屏障指令。如果改變鎖釋放的語(yǔ)義
—— 釋放時(shí)執(zhí)行一個(gè)雙向的內(nèi)存屏障 —— 將會(huì)帶來性能損失。

更多不起作用的“修復(fù)”

可以做些事情迫使寫操作的時(shí)候執(zhí)行一個(gè)雙向的內(nèi)存屏障。這是非常重量級(jí)和低效的,且?guī)缀蹩梢钥隙ㄒ坏㎎ava內(nèi)存模型修改就不能正確工作了。不要這么用。如果對(duì)此感興趣,我在另一個(gè)網(wǎng)頁(yè)上描述了這種技術(shù)。不要使用它。

但是,即使初始化helper對(duì)象的線程用了雙向的內(nèi)存屏障,仍然不起作用。

問題在于,在某些系統(tǒng)上,看到helper字段是非null的線程也需要執(zhí)行內(nèi)存屏障。

為何?因?yàn)樘幚砥饔凶约罕镜氐膶?duì)內(nèi)存的緩存拷貝。在有些處理器上,除非處理器執(zhí)行一個(gè)cache coherence指令(即,一個(gè)內(nèi)存屏障),否則讀操作可能從過期的本地緩存拷貝中取值,即使其它處理器使用了內(nèi)存屏障將它們的寫操作寫回了內(nèi)存。

我開了另一個(gè)頁(yè)面來討論這在Alpha處理器上是如何發(fā)生的。

值得費(fèi)這么大勁嗎?

對(duì)于大部分應(yīng)用來說,將getHelper()變成同步方法的代價(jià)并不高。只有當(dāng)你知道這確實(shí)造成了很大的應(yīng)用開銷時(shí)才應(yīng)該考慮這種細(xì)節(jié)的優(yōu)化。

通常,更高級(jí)別的技巧,如,使用內(nèi)部的歸并排序,而不是交換排序(見SPECJVM DB的基準(zhǔn)),帶來的影響更大。

讓靜態(tài)單例生效

如果你要?jiǎng)?chuàng)建的是static單例對(duì)象(即,只會(huì)創(chuàng)建一個(gè)Helper對(duì)象),這里有個(gè)簡(jiǎn)單優(yōu)雅的解決方案。

只需將singleton變量作為另一個(gè)類的靜態(tài)字段。Java的語(yǔ)義保證該字段被引用前是不會(huì)被初始化的,且任一訪問該字段的線程都會(huì)看到由初始化該字段所引發(fā)的所有寫操作。

class HelperSingleton {
    static Helper singleton = new Helper();
}
對(duì)32位的基本類型變量DCL是有效的

雖然DCL模式不能用于對(duì)象引用,但可以用于32位的基本類型變量。注意,DCL也不能用于對(duì)long和double類型的基本變量,因?yàn)椴荒鼙WC未同步的64位基本變量的讀寫是原子操作。

// Correct Double-Checked Locking for 32-bit primitives
class Foo { 
  private int cachedHashCode = 0;
  public int hashCode() {
    int h = cachedHashCode;
    if (h == 0) 
    synchronized(this) {
      if (cachedHashCode != 0) return cachedHashCode;
      h = computeHashCode();
      cachedHashCode = h;
      }
    return h;
    }
  // other functions and members...
}

事實(shí)上,如果computeHashCode方法總是返回相同的結(jié)果且沒有其它附屬作用時(shí)(即,computeHashCode是個(gè)冪等方法),甚至可以消除這里的所有同步。

// Lazy initialization 32-bit primitives
// Thread-safe if computeHashCode is idempotent
class Foo { 
  private int cachedHashCode = 0;
  public int hashCode() {
    int h = cachedHashCode;
    if (h == 0) {
      h = computeHashCode();
      cachedHashCode = h;
      }
    return h;
    }
  // other functions and members...
}
用顯式的內(nèi)存屏障使DCL有效

如果有顯式的內(nèi)存屏障指令可用,則有可能使DCL生效。例如,如果你用的是C++,可以參考來自Doug
Schmidt等人所著書中的代碼:

// C++ implementation with explicit memory barriers
// Should work on any platform, including DEC Alphas
// From "Patterns for Concurrent and Distributed Objects",
// by Doug Schmidt
template  TYPE *
Singleton::instance (void) {
    // First check
    TYPE* tmp = instance_;
    // Insert the CPU-specific memory barrier instruction
    // to synchronize the cache lines on multi-processor.
    asm ("memoryBarrier");
    if (tmp == 0) {
        // Ensure serialization (guard
        // constructor acquires lock_).
        Guard guard (lock_);
        // Double check.
        tmp = instance_;
        if (tmp == 0) {
                tmp = new TYPE;
                // Insert the CPU-specific memory barrier instruction
                // to synchronize the cache lines on multi-processor.
                asm ("memoryBarrier");
                instance_ = tmp;
        }
    return tmp;
}
用線程局部存儲(chǔ)來修復(fù)DCL

Alexander Terekhov ([email protected])提出了個(gè)能實(shí)現(xiàn)DCL的巧妙的做法 ——
使用線程局部存儲(chǔ)。每個(gè)線程各自保存一個(gè)flag來表示該線程是否執(zhí)行了同步。

class Foo {
 /** If perThreadInstance.get() returns a non-null value, this thread
    has done synchronization needed to see initialization
    of helper */
     private final ThreadLocal perThreadInstance = new ThreadLocal();
     private Helper helper = null;
     public Helper getHelper() {
         if (perThreadInstance.get() == null) createHelper();
         return helper;
     }
     private final void createHelper() {
         synchronized(this) {
             if (helper == null)
                 helper = new Helper();
         }
     // Any non-null value would do as the argument here
         perThreadInstance.set(perThreadInstance);
     }
}

這種方式的性能嚴(yán)重依賴于所使用的JDK實(shí)現(xiàn)。在Sun 1.2的實(shí)現(xiàn)中,ThreadLocal是非常慢的。在1.3中變得更快了,期望能在1.4上更上一個(gè)臺(tái)階。Doug Lea分析了一些延遲初始化技術(shù)實(shí)現(xiàn)的性能

在新的Java內(nèi)存模型下

JDK5使用了新的Java內(nèi)存模型和線程規(guī)范。

用volatile修復(fù)DCL

JDK5以及后續(xù)版本擴(kuò)展了volatile語(yǔ)義,不再允許volatile寫操作與其前面的讀寫操作重排序,也不允許volatile讀操作與其后面的讀寫操作重排序。更多詳細(xì)信息見Jeremy Manson的博客。

// Works with acquire/release semantics for volatile
// Broken under current semantics for volatile
class Foo {
    private volatile Helper helper = null;
    public Helper getHelper() {
        if (helper == null) {
            synchronized(this) {
                if (helper == null)
                    helper = new Helper();
            }
        }
        return helper;
    }
}
不可變對(duì)象的DCL

如果Helper是個(gè)不可變對(duì)象,那么Helper中的所有字段都是final的,那么不使用volatile也能使DCL生效。主要是因?yàn)橹赶虿豢勺儗?duì)象的引用應(yīng)該表現(xiàn)出形如int和float一樣的行為;讀寫不可變對(duì)象的引用是原子操作。

原文 Double Checked Locking
翻譯 丁一
via ifeve

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/64060.html

相關(guān)文章

  • 為什么雙重檢查鎖模式需要 volatile ?

    摘要:注意,禁止指令重排序在之后才被修復(fù)使用局部變量?jī)?yōu)化性能重新查看中雙重檢查鎖定代碼。幫助文檔雙重檢查鎖定與延遲初始化有關(guān)雙重檢查鎖定失效的說明 雙重檢查鎖定(Double check locked)模式經(jīng)常會(huì)出現(xiàn)在一些框架源碼中,目的是為了延遲初始化變量。這個(gè)模式還可以用來創(chuàng)建單例。下面來看一個(gè) Spring 中雙重檢查鎖定的例子。 showImg(https://segmentfaul...

    geekzhou 評(píng)論0 收藏0
  • 單例模式

    摘要:構(gòu)造函數(shù)被調(diào)用或者,我們利用初始化塊,在初始化的時(shí)候就完成實(shí)例化構(gòu)造器被調(diào)用雙重檢查鎖定避免懶漢模式造成性能低下的另一個(gè)思路就是雙重檢查鎖定。 1. 什么是單例 保證一個(gè)類僅有一個(gè)實(shí)例,并提供一個(gè)訪問它的全局訪問點(diǎn)。適用于: 當(dāng)類只能有一個(gè)實(shí)例而且客戶可以從一個(gè)眾所周知的訪問點(diǎn)訪問它時(shí)。 當(dāng)這個(gè)唯一實(shí)例應(yīng)該是通過子類化可擴(kuò)展的,并且客戶應(yīng)該無需更改代碼就能使用一個(gè)擴(kuò)展的實(shí)例時(shí)。 在...

    Backache 評(píng)論0 收藏0
  • 淺談雙重檢查鎖定和延遲初始化

    摘要:非線程安全的雙重檢查鎖這里看起來很完美,但是是一個(gè)錯(cuò)誤的優(yōu)化,代碼在讀取到不為的時(shí)候,引用的對(duì)象有可能換沒有完成初始化,這樣返回的是有問題的。 在Java多線程程序中,有時(shí)需要采用延遲初始化來降低初始化類和創(chuàng)建對(duì)象的開銷,雙重檢查鎖定是常見的延遲初始化技術(shù),但它是一種錯(cuò)誤的用法 雙重檢查鎖的演進(jìn)以及問題 使用syncronized實(shí)現(xiàn) public synchronized stati...

    Shonim 評(píng)論0 收藏0
  • 雙重檢查鎖定與延遲初始化

    摘要:基于的雙重檢查鎖定的解決方案對(duì)于前面的基于雙重檢查鎖定來實(shí)現(xiàn)延遲初始化的方案指示例代碼,我們只需要做一點(diǎn)小的修改把聲明為型,就可以實(shí)現(xiàn)線程安全的延遲初始化。 雙重檢查鎖定的由來 在java程序中,有時(shí)候可能需要推遲一些高開銷的對(duì)象初始化操作,并且只有在使用這些對(duì)象時(shí)才進(jìn)行初始化。此時(shí)程序員可能會(huì)采用延遲初始化。但要正確實(shí)現(xiàn)線程安全的延遲初始化需要一些技巧,否則很容易出現(xiàn)問題。比如,下...

    yvonne 評(píng)論0 收藏0
  • 單例模式與雙重檢查鎖定(double-checked locking)

    摘要:對(duì)于而言,它執(zhí)行的是一個(gè)個(gè)指令。在指令中創(chuàng)建對(duì)象和賦值操作是分開進(jìn)行的,也就是說語(yǔ)句是分兩步執(zhí)行的。此時(shí)線程打算使用實(shí)例,卻發(fā)現(xiàn)它沒有被初始化,于是錯(cuò)誤發(fā)生了。 1.餓漢式單例 public class Singleton { private static Singleton instance = new Singleton(); ...

    yearsj 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<