无法导出模板功能

ufu*_*gun 8 c++ templates dllexport

我有一个名为"SimObject"的类:

namespace simBase
{
    class __declspec(dllexport) SimObject: public SimSomething
    {
        public:

            template <class T>
            void updateParamValue( const std::string& name, T val );
    }
}
Run Code Online (Sandbox Code Playgroud)

我有另一个名为"ITerrainDrawable"的类:

namespace simTerrain
{
    class __declspec(dllexport) ITerrainDrawable : public simBase::SimObject
    {
    }
}
Run Code Online (Sandbox Code Playgroud)

这些类位于不同的库中.SimObject在simBase中,ITerrainDrawable在simTerrain库中.即使ITerrainDrawable派生自SimObject并且我包含了simBase库,我也会收到链接错误:

未解决的外部符号

1>ITerrainDrawable.obj : error LNK2019: unresolved external symbol "public: void __thiscall simBase::SimObject::updateParamValue<float>(class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > const &,float)" (??$updateParamValue@M@SimObject@simBase@@QAEXABV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@M@Z) referenced in function "public: void __thiscall simTerrain::ITerrainDrawable::setTerrainSize(float)" (?setTerrainSize@ITerrainDrawable@simTerrain@@QAEXM@Z)
1>ITerrainDrawable.obj : error LNK2019: unresolved external symbol "public: void __thiscall simBase::SimObject::updateParamValue<class osg::Vec4f>(class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > const &,class osg::Vec4f)" (??$updateParamValue@VVec4f@osg@@@SimObject@simBase@@QAEXABV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@VVec4f@osg@@@Z) referenced in function "public: void __thiscall simTerrain::ITerrainDrawable::setSatelliteTextureBorders(class osg::Vec2f,class osg::Vec2f)" (?setSatelliteTextureBorders@ITerrainDrawable@simTerrain@@QAEXVVec2f@osg@@0@Z)
Run Code Online (Sandbox Code Playgroud)

为什么我会收到此错误?

如果我不使用模板功能但我需要它,一切正常.

如果我将此函数移动到simTerrain库它工作正常,但我不想使用重复函数,因为有许多库像simTerrain.

小智 12

C++并不真正支持单独编译模板代码 - 您需要将模板的定义放在头文件中.


Ash*_*ish 6

模板有两种编译模型:

  1. 包含编译模型(如包含头文件)
  2. 单独的编译模型(接口与实现的分离)

您可以在模板定义时使用export关键字作为第二个选项.

export template <class T>
void updateParamValue( const std::string& name, T val ) {}
Run Code Online (Sandbox Code Playgroud)

但我不确定所有编译器都支持它.


Pra*_*rav 5

模板类的完整实现必须位于该模板类的标头中.C++的ANSI/ISO标准允许使用export关键字将实现放在单独的编译单元中,但目前有任何编译器实际支持这一点.

欲了解更多信息,请阅读这一点.


Tom*_*bes 5

您需要指定(cpp 文件很好)模板专业化。这意味着您需要导出模板参数的所有组合,您将使用:

template __declspec(dllexport) void updateParamValue<int>( const std::string& name, T val );
template __declspec(dllexport) void updateParamValue<short>( const std::string& name, T val );
......
Run Code Online (Sandbox Code Playgroud)

我在 Visual Studio 2013 中使用它并且它有效。