473,569 Members | 2,406 Online
Bytes | Software Development & Data Engineering Community
+ Post

Home Posts Topics Members FAQ

Static initialization order fiasco with smart pointers

I've a class C with a smart pointer (I use boost::shared_p tr) which is
initialized in the constructor:

class C {
boost::shared_p tr<D> d;
public:
C() : d(new d()) { }
};

When the program starts class C is instantiated quite a lot. As all the
instances of C use the same empty instance of D I want them to share the
instance:

class C {
boost::shared_p tr<D> d;
static boost::shared_p tr<D> one;
public:
C() : d(one) { }
};

boost::shared_p tr<D> C::one(new d());

That would work if I wasn't using the quick_allocator for boost::shared_p tr
(a memory manager which is used when BOOST_SP_USE_QU ICK_ALLOCATOR is
defined). Now as it turns out this memory manager uses static variables.

According to the FAQ we initialize the static member now in a method:

class C {
boost::shared_p tr<D> d;
static boost::shared_p tr<D> one;
public:
C() : d(one) { }
static Init() { one.reset(new d()); }
};

boost::shared_p tr<D> C::one;

Now everything works until the program ends. If the static variables in the
memory manager of boost::shared_p tr are destroyed first the destructor of
the static smart pointer in class C will make a call to a non-existing
memory manager.

The only solution I see is freeing the static smart pointer when the last
instance of class C is destroyed (then noone needs the instance of D in the
smart pointer anymore). The destructor uses the counter of the smart pointer
to check when it should be freed:

class C {
boost::shared_p tr<D> d;
static boost::shared_p tr<D> one;
public:
C() : d(one) { }
static Init() { one.reset(new d()); }
~C() { if (one.unique()) one.reset(); }
};

boost::shared_p tr<D> C::one;

This is however no general solution (eg. if you destroy all instances of C
and then create a new one in the middle of the program the smart pointer is
empty). Is there a better idea how to handle this?

Boris
Apr 13 '06 #1
5 7790
Boris wrote:
[...] This is however no general solution (eg. if you destroy all
instances
of C and then create a new one in the middle of the program the smart
pointer is empty). Is there a better idea how to handle this?


Alright, about three seconds after clicking on Send I suddenly understood
that the smart pointer must be created dynamically:

class C {
boost::shared_p tr<D> d;
static boost::shared_p tr<D> *one;
public:
C() : d(*one) { }
static Init() { one = new boost::shared_p tr<D>(new d()); }
};

boost::shared_p tr<D> *C::one;

Sometimes it helps to send a question just to find the answer yourself. :)

Boris
Apr 13 '06 #2
Boris wrote:
Boris wrote:
[...] This is however no general solution (eg. if you destroy all
instances
of C and then create a new one in the middle of the program the smart
pointer is empty). Is there a better idea how to handle this?
Alright, about three seconds after clicking on Send I suddenly
understood that the smart pointer must be created dynamically:

class C {
boost::shared_p tr<D> d;
static boost::shared_p tr<D> *one;
public:
C() : d(*one) { }
static Init() { one = new boost::shared_p tr<D>(new d()); }


static *WHAT* Init()? Did you mean to say

static void Init()

???
};

boost::shared_p tr<D> *C::one;
This is initialised to a null pointer (since it's "static"). Is that
really what you wanted? As soon as I try creating an instance of 'C',
I get undefined behaviour because a null pointer is dereferenced...
Sometimes it helps to send a question just to find the answer
yourself. :)


No doubt. Did you actually find the answer?

V
--
Please remove capital 'A's when replying by e-mail
I do not respond to top-posted replies, please don't ask
Apr 13 '06 #3
Victor Bazarov wrote:
[...] static *WHAT* Init()? Did you mean to say

static void Init()

???


Sorry, I forgot the return type.
boost::shared_p tr<D> *C::one;


This is initialised to a null pointer (since it's "static"). Is that
really what you wanted? As soon as I try creating an instance of 'C',
I get undefined behaviour because a null pointer is dereferenced...


Read my original message again and then the FAQ. Here's the direct link:
http://www.parashift.com/c++-faq-lit...html#faq-10.14
Sometimes it helps to send a question just to find the answer
yourself. :)


No doubt. Did you actually find the answer?


Guess for what Init() is used for you corrected above?

Boris
Apr 13 '06 #4
Boris wrote:
Victor Bazarov wrote:
[...] static *WHAT* Init()? Did you mean to say

static void Init()

???


Sorry, I forgot the return type.
boost::shared_p tr<D> *C::one;


This is initialised to a null pointer (since it's "static"). Is that
really what you wanted? As soon as I try creating an instance of
'C', I get undefined behaviour because a null pointer is
dereferenced...


Read my original message again and then the FAQ. Here's the direct
link: http://www.parashift.com/c++-faq-lit...html#faq-10.14
Sometimes it helps to send a question just to find the answer
yourself. :)


No doubt. Did you actually find the answer?


Guess for what Init() is used for you corrected above?


I don't see it called anywhere.

V
--
Please remove capital As from my address when replying by mail
Apr 14 '06 #5

Boris wrote:
Boris wrote:
[...] This is however no general solution (eg. if you destroy all
instances
of C and then create a new one in the middle of the program the smart
pointer is empty). Is there a better idea how to handle this?


Alright, about three seconds after clicking on Send I suddenly understood
that the smart pointer must be created dynamically:

class C {
boost::shared_p tr<D> d;
static boost::shared_p tr<D> *one;
public:
C() : d(*one) { }
static Init() { one = new boost::shared_p tr<D>(new d()); }
};

boost::shared_p tr<D> *C::one;

Sometimes it helps to send a question just to find the answer yourself. :)

Boris


Try the following code:

#include <iostream>
#include <boost/shared_ptr.hpp>

struct D {
int x;
D():x(3){}
};

class C {
boost::shared_p tr<D> d;
public:
C(): d(Init()) {}
static boost::shared_p tr<D> Init(){
static boost::shared_p tr<D> *one;
if(!one)
one = new boost::shared_p tr<D>(new D());
return *one;
}
void print(){
std::cout << "D::x = " << d->x << std::endl;
}
};

int main(void){

C c;
c.print();
}

Apr 19 '06 #6

This thread has been closed and replies have been disabled. Please start a new discussion.

Similar topics

1
4611
by: Qin Chen | last post by:
I will present very long code, hope someone will read it all, and teach me something like tom_usenet. This question comes to me when i read <<Think in C++>> 2nd, chapter 10 , name control, section "Static initialization dependency". There is a example to show how to solve the prolem involved with a technique first poineered by Jerry Schwarz...
4
2184
by: Bret Pehrson | last post by:
I just stumbled across the following problem: //.h class Masses { static double mass1; static double mass2; static double mass3; };
4
1790
by: ma740988 | last post by:
Referencing source snippet below, the actual contruction of the foo objects is done in a class. In that regard, I chose methods, class1_construct and class2_construct for demonstration purposes. That aside, I encountered source akin to what's shown below today and I was almost convinced the source is wrought with trouble. On second thought it...
14
2557
by: Jeroen | last post by:
Hi all, I've got a question about writing a library. Let me characterize that library by the following: * there is a class A which is available to the user * there is a class B that is used in severel 'underwater operations' * there is a list which stores objects of class B There are several issues I'm not sure about:
1
3516
by: Sandro Bosio | last post by:
Hello everybody, my first message on this forum. I tried to solve my issue by reading other similar posts, but I didn't succeed. And forgive me if this mail is so long. I'm trying to achieve the following (with incomplete succes): I want in a given namespace Parameters a list of "initializers" (which are objects derived from a simple interface...
10
4187
by: n.torrey.pines | last post by:
Are global variables (and const's) guaranteed to be initialized before static class members (and methods) ? const int x = 19907; int get_x() { return x; } // another compilation unit: int get_x();
6
2157
by: r.z. | last post by:
They should be initialized before any instance is created. I have no idea in which file, in which place should I put their initialization code to be sure they are initialize only once, before any instance is created. I cannot visualize the control flow of an executing program when the code is split in multiple files.
3
5838
by: Steve Folly | last post by:
Hi, I had a problem in my code recently which turned out to be the 'the "static initialization order fiasco"' problem (<http://www.parashift.com/c++-faq-lite/ctors.html#faq-10.12>) The FAQ section describes a solution using methods returning references to static objects. But consider:
20
6074
by: JohnQ | last post by:
The way I understand the startup of a C++ program is: A.) The stuff that happens before the entry point. B.) The stuff that happens between the entry point and the calling of main(). C.) main(). So, if the above is OK, does static initialization occur during A or B? What happens during A?
0
7618
by: Hystou | last post by:
Most computers default to English, but sometimes we require a different language, especially when relocating. Forgot to request a specific language before your computer shipped? No problem! You can effortlessly switch the default language on Windows 10 without reinstalling. I'll walk you through it. First, let's disable language...
0
7926
Oralloy
by: Oralloy | last post by:
Hello folks, I am unable to find appropriate documentation on the type promotion of bit-fields when using the generalised comparison operator "<=>". The problem is that using the GNU compilers, it seems that the internal comparison operator "<=>" tries to promote arguments from unsigned to signed. This is as boiled down as I can make it. ...
1
7678
by: Hystou | last post by:
Overview: Windows 11 and 10 have less user interface control over operating system update behaviour than previous versions of Windows. In Windows 11 and 10, there is no way to turn off the Windows Update option using the Control Panel or Settings app; it automatically checks for updates and installs any it finds, whether you like it or not. For...
0
7982
tracyyun
by: tracyyun | last post by:
Dear forum friends, With the development of smart home technology, a variety of wireless communication protocols have appeared on the market, such as Zigbee, Z-Wave, Wi-Fi, Bluetooth, etc. Each protocol has its own unique characteristics and advantages, but as a user who is planning to build a smart home system, I am a bit confused by the...
1
5514
isladogs
by: isladogs | last post by:
The next Access Europe User Group meeting will be on Wednesday 1 May 2024 starting at 18:00 UK time (6PM UTC+1) and finishing by 19:30 (7.30PM). In this session, we are pleased to welcome a new presenter, Adolph Dupré who will be discussing some powerful techniques for using class modules. He will explain when you may want to use classes...
0
5222
by: conductexam | last post by:
I have .net C# application in which I am extracting data from word file and save it in database particularly. To store word all data as it is I am converting the whole word file firstly in HTML and then checking html paragraph one by one. At the time of converting from word file to html my equations which are in the word document file was convert...
0
3656
by: TSSRALBI | last post by:
Hello I'm a network technician in training and I need your help. I am currently learning how to create and manage the different types of VPNs and I have a question about LAN-to-LAN VPNs. The last exercise I practiced was to create a LAN-to-LAN VPN between two Pfsense firewalls, by using IPSEC protocols. I succeeded, with both firewalls in...
1
2116
by: 6302768590 | last post by:
Hai team i want code for transfer the data from one system to another through IP address by using C# our system has to for every 5mins then we have to update the data what the data is updated we have to send another system
1
1226
muto222
by: muto222 | last post by:
How can i add a mobile payment intergratation into php mysql website.

By using Bytes.com and it's services, you agree to our Privacy Policy and Terms of Use.

To disable or enable advertisements and analytics tracking please visit the manage ads & tracking page.