全局静态变量的实例化行为

3
我的问题很简单,但答案可能不是。
在C++中(在Win7上使用Intel C++ 13.1编译器),全局静态变量是否总是在执行main()之前实例化?如果不是,那么它是否取决于编译选项(比如/Ox)?
如果它们在DLL中声明和定义,情况是否相同?
以下是一个案例:
我有这样的一些代码:
// in DLL.h
class MyClass
{
public:
    MyClass();
};

static MyClass *sgMyClassPtr;

并且

// in DLL.cpp
MyClass *sgMyClassPtr = new MyClass;

MyClass::MyClass()
{
    // Code to execute here
}

请注意,我省略了导出声明,但它已被正确导出。
从我的主应用程序代码来看,当我运行它时似乎并没有始终执行MyClass :: MyClass()。我真的不理解,但似乎如果 DLL 还没有被加载或静态未被正确实例化,则会出现这种情况。请注意,没有线程,并且每个调用都是同步的(至少在我的代码中是如此!)
如果您有任何想法或建议,将不胜感激。谢谢!
更新1
也许更容易告诉您我想要得到什么而不是我做了什么...
我想要一个变量,在 DLL 加载时自动实例化。该变量将由应用程序(.exe)中的单例注册(存储在 std::set 中的 ptr)。应用程序单例不知道 DLL,但 DLL 知道应用程序单例。因此,在 DLL 加载时,我希望该变量立即实例化,然后在应用程序单例中注册自己。这就是为什么我在 DLL 中声明了静态变量并在那里实例化它的原因。注册在 cTor 中完成。
我的最初问题是:静态实例化是否会在 DLL 加载时立即发生,或者可能会延迟?我之所以问这个问题是因为有时候我会观察到奇怪的行为,看起来像是异步问题...???

1
你是不是要说 extern MyClass* sgMyClassPtr 而不是 static *MyClass sgMyClassPtr - David
不,我的意思是静态的MyClass*。该变量在DLL中声明和实例化。 - dom_beau
这是一篇解释变量如何初始化的帖子:https://dev59.com/nGMm5IYBdhLWcg3wMs7R - Gabriel
1个回答

0

静态初始化发生在 DLL 被加载时,但根据链接器选项,DLL 可以被按需加载。请注意,如果您将类包含在 DLL 和主程序中,但不从 DLL 导出它,则会得到两份代码副本,并且可能会有两份(类)静态变量的副本。因此,当另一个已经初始化时,您可能会因为其中一个副本未初始化而感到困惑。

但请确保您首先了解关于延迟加载 DLL 的链接器选项。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接