Freigeben über


How to utilize double check locking in C# correctly

The major benefit of double check locking is to reduce the overhead of acquiring a lock by first testing the locking in an unsafe manner under a multi-threaded environment. It is widely used in the scenarios such as lazy initialization. Typical pseudo code would be like followings:

IF (VAR_X meets certain conditions)

{

ACQUIRE_LOCK();

       IF (VAR_X meets certain conditions)

{

                   VAR_X = NEW_VAR_X;

                   DO_SOMETHING_ELSE();

}

RELEASE_LOCK();

}

USE(VAR_X);

Unfortunately, due to the aggressive optimization of modern compiler, thread A might possibly access partial result of VAR_X before thread B actually completes updating it. As a result, the app will probably crash.

In C#, to ensure the correctness, we should declare VAR_X as volatile to ensure the assignment to the VAR_X completes before it can be accessed. I am working on a sample to demonstrate this; Can anyone share working code before I post my version out?

Singleton implementation in msdn: https://msdn.microsoft.com/en-us/library/ms998558.aspx

Comments

  • Anonymous
    September 14, 2008
    PingBack from http://www.easycoded.com/how-to-utilize-double-check-locking-in-c-correctly/
  • Anonymous
    September 14, 2008
    The annoying part of it is that C# disallows "volatile" for local variables, even though they can perfectly well be accessed concurrently.
  • Anonymous
    September 14, 2008
    http://blogs.msdn.com/brada/archive/2004/05/12/volatile-and-memorybarrier.aspxYou can do a MemoryBarrier directly which can be more efficient overall...
  • Anonymous
    September 15, 2008
    Note that your example is incorrect. The new_var_x needs to be stored to a local, then fully initialized, and only then can it be stored to VAR_X. Otherwise, you have the problem of VAR_X containing a partially initialized object.
  • Anonymous
    September 20, 2008
    Thanks for the comment. It is a great post containing informative discussion. Although people agreed on volatile implementation, I find I have troubles in figuring out which one is the correct implementations for MemoryBarrier. Specifically do we need MemoryBarrier for both reads and writes?Thoughts?
  • Anonymous
    September 20, 2008
    That is why I call it pseudo code. :-)