Consider the following scenario. We have a C++ function with a static local variable:
void function()
{
static int variable = obtain();
//blahblablah
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
}
Some lateral dodges you can try that might solve your underlying problem:
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.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).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
}
(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.