DLL导出函数名称改编的解决方法

1.DLL编译后导出函数名称改编

 

在编写一个DLL后,为了能被别的程序调用,需要将被使用的函数导出;

但是一般的编译器都会将到处函数名称改编;

例如:在VC中新建一个空的win32 dll工程,然后添加下面的文件;

[cpp] view plain copy
  1. //dll1.h  
  2. #ifdef DLL_API _declspec(dllexport)   
  3. #else  
  4. #define DLL_API _declspec(dllexport)  
  5. #endif  
  6.   
  7. DLL_API int add(int a, int b);  

[cpp] view plain copy
  1. //dll1.cpp  
  2.   
  3. #include "dll1.h"  
  4.   
  5. DLL_API int add(int a, int b)  
  6. {  
  7.   return a + b;  
  8. }  

 

编译以后,得到dll1.dll与dll1.lib;使用Dumpbin.exe或者是Depends工具查看发现,

dll1.dll中的到处函数名称为:?add@@YAHH@Z ;这个新的函数名称是C++编译器对add函数的名称进行了改编,

而且不同的编译器的改编规则也不一样,这就导致在通过add函数名对函数进行调用时无法找到add函数,

因为此时add的函数名称已经被改编;

 

2.限定导出函数名称

 

为了解决C与C++能在不同编译器之间正常调用DLL,所以我们希望DLL在编译过程中不要对函数名称进行改编;

我们可以在定义导出函数时,加上限定符 extern "C"

我们把上面的例子修改下:

[cpp] view plain copy
  1. //dll1.h  
  2. #ifdef DLL_API extern "C" _declspec(dllexport)   
  3. #else  
  4. #define DLL_API extern "C" _declspec(dllexport)  
  5. #endif  
  6.   
  7. DLL_API int add(int a, int b);  

 

[cpp] view plain copy
  1. //dll1.cpp  
  2. #include "dll1.h"  
  3.   
  4. DLL_API int add(int a, int b)  
  5. {  
  6.   return a + b;  
  7. }  

我们再次编译得到dll1.dll,通过工具查看其到导出函数发现,此时的add函数的导出名称仍然是add;

这样我们就可以在其他编译器上直接通过add调用该函数了;

 

3.__stdcall关键字将使限定无效

如果我们在第二个的基础上给函数加上__stdcall关键字,导出函数的名称将仍然被改编;

如果没有添加__stdcall关键字,那么函数调用约定为C调用约定。如果加了__stdcall标准调用约定,

就是WINAPI调用约定,也就是pascal调用约定,这种约定与C调用约定不一样。

[cpp] view plain copy
  1. //dll1.h  
  2. #ifdef DLL_API extern "C" _declspec(dllexport)   
  3. #else  
  4. #define DLL_API extern "C" _declspec(dllexport)  
  5. #endif  
  6.   
  7. DLL_API int __stdcall add(int a, int b);  

 

[cpp] view plain copy
  1. //dll1.cpp  
  2. #include "dll1.h"  
  3.   
  4. DLL_API __stdcall int add(int a, int b)  
  5. {  
  6.   return a + b;  
  7. }  

重新编译,然后通过工具查看DLL的导出函数,发现名称为: _add@8;

也就是说如果函数的调用约定发生变化,即使在声明时使用了 extern "C"限定符,函数名称仍然会

改编;

C语言与Delphi的调用约定是不一样的,Delphi使用的是pascal调用约定,如果我们要用C写一个DLL供Delphi使用,

那么在导出函数时应指定其使用标准的函数调用约定,但此时 导出函数名称就会被改编;

 

在这种情况下,我们需要通过一个称为模块定义文件(DEF)的方法解决名称被改编的问题;

 

在上面例子的基础上,我们给这个工程添加一个后缀为def的文件dll1.def;然后添加如下代码:

[c-sharp] view plain copy
  1. //dll1.def  
  2.   
  3. LIBRARY "dll1"  
  4.   
  5. EXPORTS  
  6. add  

此文件中LIBRARY指定动态链接库的内部名称,该名称与生成的动态链接库名称要匹配;

EXPORTS下面就是要导出的函数;

如果导出的函数名称与源文件中的函数名称不一样可以通过下面的语法指定导出函数名称:

entryname = internalname

 

编译时,编译器会按照def中指定的函数名称导出函数;

 

重新编译,通过工具查看dll1.dll中的导出函数为add了;

阅读更多
换一批

没有更多推荐了,返回首页