在类库项目中创建模板类C++

Creating template classes in a Class Library project C++

本文关键字:C++ 建模 创建 类库 项目      更新时间:2023-10-16

我有一个C++类库项目,通常被其他C++项目使用。为了能够在我的类库项目中使用类,我编写了一个头文件,如下所示的示例

#pragma once
#ifdef MYLIB
# define MYLIB_EXPORT __declspec(dllexport)
#else
# define MYLIB_EXPORT __declspec(dllimport)
#endif

没问题,直到我想在我的类库项目中创建一个模板类。问题是我无法导出模板类。

MyClass.h

template<class T>
class MYLIB_EXPORT MyClass
{
    void myMethod();
    // ...
}
template<class T>
void MyClass::myMethod()
{
    // ...
}

在这种情况下,我收到编译错误,说"不允许定义dllimport函数"。我知道是什么导致了问题,我理解它。使用我的类库项目的其他项目将 MYLIB_EXPORT 关键字转换为 __declspec(dllimport)。因此,他们期望在 DLL 中定义 MyClass 的方法。但是,编译器随后会看到标头中的定义。

如何克服这种情况并能够导出在类库项目中定义的模板类?

未实例化的模板不能直接编译 - 它们是代码生成器,因此它们实际上只有在实例化时才被转换为二进制指令;因此,您不能"以二进制形式"导出模板,就好像它是"常规"函数/类一样(另一方面,至少在理论上您可以导出模板的实例化)。

长话短说:只需将模板保留在标题中即可由库客户端包含。

请注意,这就是将模板保留在标头中的确切原因,并且通常不会将它们的实现分离到.cpp文件中。

只需删除模板类上的 MYLIB_EXPORT 语句即可。然后,您可以在类外部定义类函数(但仍在*.h*.hpp头文件中)。

MyClass.h

    template <typename T>
    class MyClass    // MYLIB_EXPORT removed
    {
        void myMethod();
        // ...
    };
    template <typename T>
    void MyClass<T>::myMethod()
    {
        // ...
    }

我遇到了这个问题。很长一段时间后,我意识到删除MYLIB_EXPORT修复了它。希望这个答案能为其他人节省时间:-)