How to address thread-safety of service data used for maintaining static local variables in C++?

后端 未结 4 595
隐瞒了意图╮
隐瞒了意图╮ 2020-12-30 15:31

Consider the following scenario. We have a C++ function with a static local variable:

void function()
{
    static int variable = obtain();
    //blahblablah         


        
相关标签:
4条回答
  • 2020-12-30 15:49

    C++ says that your static variable should only be initialized once - however C++ doesn't deal with threads(yet).

    gcc(atleast on *nix systems) does the proper magic to safely guard multiple threads initializing such a static variable. According to http://social.msdn.microsoft.com/Forums/en/vcgeneral/thread/12f8e2c7-d94d-4904-8c79-a0b0d1088e0b , msvc does not - and in such a case you'll have to lock the initialization yourself.

    Guarding the initialization with a critical section should protect all this - i.e. your functionThreadSafe() is ok - (unless obtain() itself calls functionThreadSafe()

    http://blogs.msdn.com/b/oldnewthing/archive/2004/03/08/85901.aspx is worth a read in this regard.

    Personally, to avoid surprises I'd try to rewrite this so you can initialize variable yourself, once, before you create any threads - e.g.

    static int variable = 0;
    void init_variable() //call this once, at the start of main()
    {
      variable = obtain();
    }
    
    void function() 
    {
      //use variable, lock if you write to it
    }
    
    0 讨论(0)
  • 2020-12-30 16:01

    Some lateral dodges you can try that might solve your underlying problem:

    • you could make int variable be a thread-local static, if the different threads don't actually need to share the value of this variable or pass data to each other through it.
    • for an int on x86, you can use an atomic read/write such as by InterlockedCompareExchange() or its equivalent on your platform. This lets multiple threads safely access the variable without locks. It only works for hardware-native atomic types, though (eg, 32-bit and 64-bit words). You will also have to figure out what to do if two threads want to write to the variable at the same time (ie, one will discover that the other has written to it when it performs the compare-and-swap op).
    0 讨论(0)
  • 2020-12-30 16:02

    To avoid the locking in any case, you can go with this:

    void functionThreadSafe()
    {
        static int *variable = 0;
        if (variable == 0)
        {
           CriticalSectionLockClass lock( criticalSection );
           // Double check for initialization in different thread
           if (variable == 0)
           {
              variable = new int(obtain());
           }
        }
        //blahblablah
    }
    
    0 讨论(0)
  • 2020-12-30 16:12

    (I have post this on another question, but it is also an answer to this one)

    Here is my take (if really you can't initialize it before threads are launched):

    I've seen (and used) something like this to protect static initialization, using boost::once

    #include <boost/thread/once.hpp>
    
    boost::once_flag flag;
    
    // get thingy
    const Thingy & get()
    {
        static Thingy thingy;
    
        return thingy;
    }
    
    // create function
    void create()
    {
         get();
    }
    
    void use()
    {
        // Ensure only one thread get to create first before all other
        boost::call_once( &create, flag );
    
        // get a constructed thingy
        const Thingy & thingy = get(); 
    
        // use it
        thingy.etc..()          
    }
    

    In my understanding, this way all threads wait on boost::call_once except one that will create the static variable. It will be created only once and then will never be called again. And then you have no lock any more.

    0 讨论(0)
提交回复
热议问题