原谅转载自:http://lukas06.blog.sohu.com/94010246.html
C++编译器在生成DLL时,会对导出函数进行名字改编,并且不同的编译器使用的改编规则不一样,因此改编的名字后的名字是不一样的。因此,如果利用不同的分别生成DLL文件和访问DLL文件的客户端,那么后者在访问该DLL文件的时候就会出现问题。例如:使用C++编写了一个DLL,而使用C语言编写的客户端进行访问就会出现问题。由于C++编译器已经对该导出函数名字进行了改编,所以用C语言编写的客户端就找不到DLL的导出函数。这就是DLL导出函数的名字改编问题。
如果希望动态链接库文件在编译时,导出函数的名称不要发生改变,那么在定义导出函数时,需要加上限定符:extern "C"。注意:双引号中的“C”一定要大写。
例如:Dll1.h头文件
#ifdef DLL1_API
#else
#define DLL1_API extern "C" _declspec(dllimport)
#endif
DLL1_API int add(int a,int b);
DLL1_API int subtract(int a,int b);
Dll1.cpp源文件
#define DLL1_API extern "C" _declspec(dllexport)
#include "Dll1.h"
int add(int a,int b)
{
return a+b;
}
int subtract(int a,int b)
{
return a-b;
}
这样利用dumpbin工具可以查看Dll2.dll的导出函数,>dumpbin -exports Dll2.dll,可以发现名字没有被改编。
利用限定符extern "C"可以解决C++和C语言之间相互调用时函数命名的问题。但是这种方法有一个缺陷:就是不能用于导出一个类的成员函数,而只能用于导出全局函数这种情况。
但是还有一个问题是,如果使用了标准调用约定,也就是pascal调用约定,WINAPI调用约定:_stdcall,此时即使使用了extern "C",仍然会出现导出函数名字被改编的问题。例如:使用C语言编写一个DLL文件,而客户端使用Delphi进行编写,那么在编写导出函数时,应该指定其使用标准的函数调用约定。此时,就会出现问题,即C语言编写的DLL文件的导出函数发生了名字改编。在这种情况下,可以使用一个称为模块定义文件(DEF)的方式解决名字改编问题。
例如:Dll2.def
LIBRARY
EXPORTS
add
subtract
如果想使用与源文件中定义的不一样的函数名,可以按照以下语法:entryname=internalname
其中,entryname是要导出的符号名,而internalname是DLL中将要导出的函数名。
示例:
LIBRARY "VerifyLocalResType"
EXPORTS
VerifyDDSSize
VerifyDDSType
VerifyTGASize
VerifyTGAType