如何在Dll中导出STL类

如何在Dll中导出STL类

简介:本文详述在DLL中导出stl类及包含stl的类的方法。例子源码

Dll无法直接导出泛型模板(generalized template),因此,如果要导出stl类,则模板必须先实例化(instantiated)。另外,如果导出的STL类使用了其他STL类,那么这些其他类必须同时被导出。目前stl中唯一能够被导出的容器是vector,其他容器(如map、set、queue、list、deque)都因包含嵌套类而不能被导出。

导出STL类的步骤:

在Dll和exe文件中,用同样版本的c运行库链接。譬如都用Msvcrt.lib(release)链接或都用Msvcrtd.lib(debug)链接。

Dll中,用__declspec(dllexport)导出模板类的实例。

在exe文件中,用__declspec(dllimport)和extern关键字从Dll中导入 stl类。

需要注意的是:当导出一个以自定义类为模板参数的stl容器,必须为这个自定义类型定义<和==运算符。譬如,如果要导出std::vector<CPerson>类,则必须为CPerson添加<和==运算符。如下: //导出stl类 std::vector<CPerson>
class CPerson
{
public:
  int m_nAge;
  char m_strName[40];
public:
  bool operator < (const CPerson& c) const
  {
    return true;
  }
  bool operator == (const CPerson& c) const
  {
    return true;
  }
};
EXPIMP_TEMPLATE template class VECDLL_API std::vector<CPerson> //显示实例化模板类
VECDLL_API int fnVecDll(std::vector<CPerson>& vecPer); //导出函数

定义这两个运算符的原因是:所有stl容器都有“比较”成员函数,这些成员函数需要调用自定义类型的<和==运算符。通常情况下,由于没有使用这些成员函数,所以它们没有被实例化,所以我们使用时一般就不需要为CPerson定义这两个运算符。然而,当显示实例化此容器类时,它所有的成员函数都需实例化,包括它的“比较”成员函数,所以这时必须实现CPerson的<和==运算符。如果CPerson并不在乎<和==的意义,我们可以像上面代码所示通过简单返回true来实现它们。

导出一个“数据成员包含stl对象”的类。方法与上类似。如下代码所示:

EXPIMP_TEMPLATE template class VECDLL_API std::vector<int>  //显示实例化std::vector<int>
class VECDLL_API CContainer
{
public:
  std::vector<int> m_vecNum;
};

完整例子请查看实例代码。

本文配套源码