__declspec(dllexport)和__declspec(dllimport) (——declspec方法创建dll的方法已验证ok)

转载:https://www.cnblogs.com/chengbing2011/p/4084125.html

__declspec(dllimport)和__declspec(dllexport)经常是成对的,在动态链接库中__declspec(dllexport)导出dll中的成员,__declspec(dllimport)导入外部dll中的成员。

但是有时候不使用dllimport和dllexport也能实现个基本的导出导入功能, 它们具有的功能如下:

1.dllimport/dllexport可以导入或者导出动态链接库中的全局变量,当然是用extern也可以实现同样的功能;

2.dllimport/dllexport的作用主要体现在导出类的静态成员方面,如果不使用它们,无法在正常是用外部dll中类的静态成员函数;

3.隐式使用dll时,不加dllimport/dllexport也是可以,使用上没什么区别,只是在生成的二进制代码上稍微有点效率损失;

4.使用dllimport/dllexport还可以体现编程语言的对称美。

以下是一个Demo实例:

创建win的dll项目空项目名为DllExport

 ====================================.h文件如下[DllExport.h]:================================================

复制代码
#ifdef DLLEXPORT_EXPORTS
#define DLLEXPORT_API __declspec(dllexport)
#else
#define DLLEXPORT_API __declspec(dllimport)
#endif

// class从dll中导出Demo
class DLLEXPORT_API CDllExport {
public:  CDllExport(void);  
};

//变量从dll中导出Demo
extern DLLEXPORT_API int nDllExport;

//函数从dll中导出Demo
DLLEXPORT_API int fnDllExport(void);
复制代码

====================================.cpp文件如下[DllExport.cpp]:==============================================

复制代码
#include "DllExport.h"

// 变量从Dll中导出Demo
DLLEXPORT_API int nDllExport=0;

// 函数从Dll中导出Demo
DLLEXPORT_API int fnDllExport(void) {  return 42; }

// 类从Dll中导出Demo,以下为导出类的构造函数
CDllExport::CDllExport() {  return; }
复制代码

创建win的exe项目名为DemoEntry

====================================调用文件如下[DemoEntry.cpp]:=============================================

复制代码
#include"DllExport.h"
#include<iostream>

int _tmain(int argc, _TCHAR* argv[])
{
    int returncnt= fnDllExport();
    std::cout<<returncnt<<std::endl;
    system("pause");
    return 0;

}
复制代码

====================================================================================================

调用Dll过程编译器设置如下:

 

同样,使用.def文件也可以导出dll中的函数供外部调用,关于.def的使用如下:

1.LIBRARY语句说明.def文件相应的DLL;

2.EXPORTS语句后列出要导出函数的名称。可以在.def文件中的导出函数名后加@n,表示要导出函数的序号为n(在进行函数调用时,这个序号将发挥其作用);

3.def文件中的注释由每个注释行开始处的分号 (;) 指定,且注释不能与语句共享一行

 以下是使用.def 文件导出DLL的Demo实例:

====================================.cpp文件如下[APIExport.cpp]:=============================================

复制代码
int nTest=1;//全局变量  

int AddAndMulti(int a,int b)  
{  
    return (a+b)*nTest;  
}  

int _stdcall SubAndMulti(int a,int b)  
{  
    return (a-b)*nTest;  
} 

int _cdecl Multiply(int a,int b)  
{  
    return a*b;  
}  
复制代码

====================================.def文件如下[DllExport.def]:==============================================

LIBRARY   DllExport_def    
EXPORTS  
AddAndMulti @3   
SubAndMulti @5  
Multiply @1  
nTest  DATA 

====================================.cpp调用代码[DllLoadEntry.cpp]:============================================

复制代码
// DllLoadEntry.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <iostream> 

using namespace std; 

//引用dll中的全局变量  
extern int _declspec(dllimport) nTest;

//声明函数,需与Dll中的函数定义一致(包括其函数调用修饰词)
int AddAndMulti(int a,int b); 

int _stdcall SubAndMulti(int a,int b);  

int _cdecl Multiply(int a,int b);  


//#pragma comment(lib,"DllExport_def.lib")
int _tmain(int argc, _TCHAR* argv[])
{
    nTest=2; 

    cout<<AddAndMulti(1,2)<<endl; 
    cout<<SubAndMulti(3,4)<<endl; 
    cout<<Multiply(4,5)<<endl;

    system("pause");

    return 0;
}
复制代码

====================================================================================================

编译环境设置如下:

 

当然也可以在代码中添加 #pragma comment(lib,"DllExport_def.lib")实现同样的效果,DLL和调用的exe在同一个目录下。

def和__declspec的使用区别:

VC++编译器会对__declspec导出的函数进行一定的变化,如下:__declspec(dllexport) int __stdcall Add()会转换为Add@0(),对于外部调用程序也是VC++时,只需要将编译产生的lib文件提供给调用者即可,而如果提供给语言程序调用,就不是那么方便了。此时,使用.def文件就可以让编译器不会对导出的函数名称进行修改。

原文地址:https://www.cnblogs.com/MCSFX/p/13074062.html