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;
};