(转)如何在Dll中导出STL类
作者:yy2better
简介:本文详述在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; };
完整例子请查看实例代码。