.lib中的C++静态变量未初始化

unc*_*all 5 c++ dll static-members static-libraries visual-studio-2010

我在VS2010中有静态库(.lib)并将它链接到我的测试项目.

lib有一个我使用下面的MACRO创建的工厂:

#define REGISTER_FACTORY(mType, my_class) \
class Factory##my_class : public CAbstractFactory\
{\
public:\
    Factory##my_class() : CAbstractFactory(mType){}\
    CBaseClass *Create()\
    { return new my_class(); }\
};\
static Factory##my_class StaticFactory##my_class;
Run Code Online (Sandbox Code Playgroud)

应该发生的是,在CAbstractFactory中,新工厂被注册mtype.但是当我检查工厂时工厂不存在.

当我使用DLL而不是.lib时,它工作正常.我的猜测是链接器不包含静态变量,因为它没有被引用,或者静态变量甚至没有包含在库中.

如何强制链接器在我的.exe中包含静态库中的所有对象.

我像这样使用宏:

// Register factory that can create CMyObject with ID=100
REGISTER_FACTORY(100, CMyObject);

class CMyObject
{
};
Run Code Online (Sandbox Code Playgroud)

CAbstractFactory如下所示:

class CAbstractFactory {
    CAbstractFactory(int id) {
        CFactory::instance().add(id, this);
    }
}
Run Code Online (Sandbox Code Playgroud)

然后在代码中的其他地方,我使用的主要.exe:

CBaseClass *pObject = CFactory::instance().Create(100);
Run Code Online (Sandbox Code Playgroud)

这将给我一个新的CMyObject.我的想法是,我有许多不同类型的对象,我有一个数据库,其中包含指定我需要的对象类型的id. 100只是一个例子.

确实,我没有直接引用.lib中的任何内容,但我希望能够使用我的工厂创建对象

CFactory类是一个简单的类,它保存所有CAbstractFactory类的寄存器(在映射中),并将create方法委托给正确的工厂.

CFactory &CFactory::Instance()
{
    static CFactory instance;
    return instance;
}
Run Code Online (Sandbox Code Playgroud)

主要问题在于我没有引用.lib中的任何内容,因为它都是通过CFactory完成的.它是有效的,如果我使它成为一个DLL并确保我添加一些引用到这个DLL,以确保它被加载.但对于.lib,我甚至添加了一个虚函数,以确保我至少有一个不包含其余代码的引用.

Arn*_*age 7

我遇到了类似的问题并通过将lib项目设置为主应用程序项目的依赖项,然后将主要项目的"链接库依赖项"和"使用库依赖项输入"设置为"是"来解决此问题.


更新:

最近我发现Visual Studio 2015现在支持/WHOLEARCHIVE链接器标志.我无法通过链接器选项找到它,但您可以将其添加为附加命令行选项.它的工作方式类似于GCC标志-whole-archive,并将其添加到目标链接器标志(而不是静态lib标志).

例如,指定/WHOLEARCHIVE:lib_name为附加链接器命令行选项,它将包含该lib中的所有符号.您也可以使用多个lib.

如果使用此选项/WHOLEARCHIVE:lib_name,则不再需要将"链接库依赖项"和"使用库依赖项输入"设置为"是".这非常适合通过CMAKE生成的解决方案.请在此处查看相关答案:https://stackoverflow.com/a/42083877/1151329