While learning from isocpp, I came across an FAQ. Which says: "What is a technique to guarantee both static initialization and static deinitialization?" And the short answer is just hinted out:
Short answer: use the Nifty Counter Idiom (but make sure you understand the non-trivial tradeoffs!).
Till now, I can't understand what does Nifty Counter Idiom mean and how it fixes the Fiasco problem. I am already aware of Construct On First Use Idiom, and its side effects while using a static object or a static pointer (wrapped in a global function returning reference)
Here's a simple example actually prone to a Fiasco problem:
//x.cpp
include "x.hpp" // struct X { X(int); void f(); };
X x{ 10 };
struct Y { Y(); };
Y::Y(){ x.f(); };
//y.cpp
include <iostream>
include "y.hpp" // struct Y { Y(); };
struct X { X(int); void f(); private: int _x; };
X::X(int i) { _x = i; };
void X::f() { std::cout << _x << std::endl; };
Y y;
From here, it's mentioned the main use of Nifty Counter Idiom
Intent: Ensure a non-local static object is initialized before its first use and destroyed only after last use of the object.
Now what I need now is, how does Nifty Counter Idiom specifically can solve both static order initialization and deinitialization in the above code, regardless of other workarounds as constinit
. Given that the above program is compiled like this:
~$ g++ x.cpp y.cpp && ./a.out
>> 10
~$ g++ y.cpp x.cpp && ./a.out
>> 0
Hmm, that is a nifty counter mechanism. It doesn't "fix" the ordering-fiasco; rather, what it is doing is using an integer counter to make it so that the (still undefined) order in which the compiler does its static-initialization and static-destruction doesn't matter.
How does it do that? Simple; no matter which
StreamInitializer
object's static-initialization happens first, the first thing theStreamInitializer()
constructor will do is incrementnifty_counter
. Sincenifty_counter
was default-initialized to zero, that means theif (nifty_counter++ == 0)
test in the constructor will return true only once, the first time it is executed, regardless of which .cpp file the initialization is being triggered from. That will cause the placement-new initialization of theStream
object to happen exactly once, as a side-effect of the firstStreamInitializer
being initialized.Similarly, each
~StreamInitializer()
destructor will decrementnifty_count
, and since there will be exactly the same number of~StreamInitializer()
destructor-calls (at the end of the program's execution) as there wereStreamInitializer()
constructor-calls (at the beginning of the program's execution), we are guaranteed that it will be the very last call to~StreamInitializer()
that decrementsnifty_counter
back to zero. That will be true regardless of the order that the compiler chooses for destruction to happen, which means that the destruction of theStream
object will occur only during the last~StreamInitializer()
call, which is what you want.The upshot is that as long as your .cpp file declares a
StreamInitializer
object first, it can safely access theStream
object, because the presence of a validStreamInitializer
object (and in particular, the execution of its constructor and destructor at the appropriate times) guarantees that theStream
object will be valid in any subsequent static-object-constructor/destructor code in that .cpp file.