好的,所以我已经阅读了几个关于这个主题的问题和文章,我觉得我理解基础知识,但我仍然遇到麻烦。
我有一个DLL,它导出一个以std :: string为成员的类。 我的主程序包含也有字符串的类,它使用DLL。
如果我在VS2010中编译DLL,我会收到以下警告:
warning C4251: 'MyClass::data' : class 'std::basic_string<_Elem,_Traits,_Ax>' needs to have dll-interface to be used by clients of class 'MyClass'
当我编译EXE时,我得到相同的警告,但没有错误,程序编译并运行。实际上,这是一个大项目,所以我得到了40个警告,我不太热衷于此。 (作为侧面观察,当使用VS2008编译时,这些警告不存在)
所以,我读到了这个警告,并引导我阅读这篇MS文章: http://support.microsoft.com/default.aspx?scid=KB;EN-US;168958 它告诉我如何从DLL导出STL模板以满足我得到的警告。
问题是,当我添加以下行以删除警告时:
EXPIMP_TEMPLATE template class DECLSPECIFIER std::allocator<char>;
EXPIMP_TEMPLATE template class DECLSPECIFIER std::basic_string< char, std::char_traits<char>, std::allocator<char> >;
DLL编译没有任何警告,但是当我编译我的EXE时,链接器会抛出一个拟合:
2>SampleDLL.lib(SampleDLL.dll) : error LNK2005: "public: __thiscall std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >::~basic_string<char,struct std::char_traits<char>,class std::allocator<char> >(void)" (??1?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@XZ) already defined in OtherClass.obj
2>SampleDLL.lib(SampleDLL.dll) : error LNK2005: "public: unsigned int __thiscall std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >::size(void)const " (?size@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIXZ) already defined in OtherClass.obj
DLL和EXE都使用相同的代码生成选项进行编译。我可以在两者或MD上使用MT,结果是一样的。
如果我遗漏了任何内容,我会将最小化的示例程序中的代码包括在内。
我的主要问题:我可以解决LNK2005错误,还是可以安全地忽略C4251警告?
编辑:所以我已经阅读了一点,看起来如果DLL类使用的std :: string是一个只能由成员函数访问的私有变量,它可能是安全地忽略警告......对此有何评论?这是朝着正确方向迈出的一步吗?
DLL代码:
#pragma once
#include <exception>
#include <string>
#ifdef SAMPLEDLL_EXPORTS
# define DECLSPECIFIER __declspec(dllexport)
# define EXPIMP_TEMPLATE
#else
# define DECLSPECIFIER __declspec(dllimport)
# define EXPIMP_TEMPLATE extern
#endif
//disable warnings on extern before template instantiation (per MS KB article)
#pragma warning (disable : 4231)
//std::basic_string depends on this allocator, so it must also be exported.
EXPIMP_TEMPLATE template class DECLSPECIFIER std::allocator<char>;
//std::string is a typedef, so you cannot export it. You must export std::basic_string
EXPIMP_TEMPLATE template class DECLSPECIFIER std::basic_string< char, std::char_traits<char>, std::allocator<char> >;
#pragma warning (default : 4231)
class DECLSPECIFIER MyClass
{
public:
std::string getData(); //returns 'data', body in CPP file
private:
std::string data;
int data2;
};
//in SampleDLL.cpp file...
std::string MyClass::getData() { return data; }
EXE代码:
#include <iostream>
#include "SampleDLL.h"
using namespace std;
void main()
{
MyClass class1;
cout << class1.getData() << endl;
}
答案 0 :(得分:3)
看起来您正在connect.microsoft.com上看到问题。
那里有一个解决方法,但似乎有点讨厌。
其他可能有用的选项:
答案 1 :(得分:2)
链接到您提供的MS文章说,某些STL类“...已经由C运行时DLL导出。因此,您无法从DLL中导出它们。”包括basic_string。并且您的链接错误表明basic_string符号“...已在OtherClass.obj中定义”。因为链接器在两个不同的位置看到两个相等的符号。
答案 2 :(得分:0)
从DLL导出STL std :: basic_string模板时,出现LNK2005错误
另请参阅Microsoft的KB 168958文章How to export an instantiation of a Standard Template Library (STL) class and a class that contains a data member that is an STL object。来自文章:
导出STL类
- 在DLL和.exe文件中,链接与C运行时的相同DLL版本。要么链接Msvcrt.lib(发布版本),要么 链接Msvcrtd.lib(调试版本)。
- 在DLL中,在模板实例化声明中提供__declspec说明符以从中导出STL类实例化 DLL。
- 在.exe文件中,在模板实例化声明中提供extern和__declspec说明符,以从中导入类。 DLL。这导致警告C4231“使用非标准扩展: 模板显式实例化之前的'extern'。“你可以忽略它 警告。
醇>
答案 3 :(得分:0)
我有一个hack可以在temp中解决这个问题
打开项目选项,单击链接器 - &gt;命令行, 在“附加选项”输入框中,键入
/FORCE:MULTIPLE
答案 4 :(得分:0)
对我而言,整个话题归结为
到目前为止,我总是为我解决问题。
不幸的是,如果您无法控制要链接的源代码,那就无法回答。