区别
他们都是DLL内的关键字,即导出与导入。他们是将DLL内部的类与函数以及数据导出与导入时使用的。
dllexport是在这些类、函数以及数据的申明的时候使用。用他表明这些东西可以被外部函数使用,即(dllexport)是把 DLL中的相关代码(类,函数,数据)暴露出来为其他应用程序使用。使用了(dllexport)关键字,相当于声明了紧接在(dllexport)关键字后面的相关内容是可以为其他程序使用的。
dllimport是在外部程序需要使用DLL内相关内容时使用的关键字。当一个外部程序要使用DLL 内部代码(类,函数,全局变量)时,只需要在程序内部使用(dllimport)关键字声明需要使用的代码就可以了,即(dllimport)关键字是在外部程序需要使用DLL内部相关内容的时候才使用。(dllimport)作用是把DLL中的相关代码插入到应用程序中。
_declspec(dllexport)与_declspec(dllimport)是相互呼应,只有在DLL内部用dllexport作了声明,才能在外部函数中用dllimport导入相关代码。
常见用法
在为方便使用,我们经常在代码中定义宏DLL_EXPORT,此宏用在需要导出的类和函数前,而此宏我们定义如下:
#ifdef DLL_EXPORTS
#define SIMPLE_CLASS_EXPORT __declspec(dllexport)
#else
#define SIMPLE_CLASS_EXPORT __declspec(dllimport)
#endif
作为动态库,在需要导出的类或函数前必须使用关键字__declspec(dllexport)声明,因此动态库需要定义宏DLL_EXPORTS(使用Vistualstudio建立动态库工程时,此宏已经定义好)。
应用程序需要使用关键字__declspec(dllimport),因此不能定义宏DLL_EXPORTS。
可以省略dllimport???
但MSDN文档里面,对于 __declspec(dllimport)的说明让人感觉有点奇怪,先来看看MSDN里面是怎么说的:
不使用 __declspec(dllimport)也能正确编译代码,但使用 __declspec(dllimport) 使编译器可以生成更好的代码。编译器之所以能够生成更好的代码,是因为它可以确定函数是否存在于 DLL 中,这使得编译器可以生成跳过间接寻址级别的代码,而这些代码通常会出现在跨DLL 边界的函数调用中。但是,必须使用 __declspec(dllimport) 才能导入 DLL 中使用的变量。
使用__declspec(dllimport)可以生成更好的代码,这点好理解,但必须使用它才能导出dll中的变量,真的是如此吗?那我们就来测试一下:
建立动态库,由SimpleClass.h和SimpleClass.cpp组成,SimpleClass.h的代码实现如下:
//file SimpleClass.h
#ifndef _SIMPLE_CLASS_H_
#define _SIMPLE_CLASS_H_
#ifdef DLL_EXPORTS
#define SIMPLE_CLASS_EXPORT__declspec(dllexport)
#else
#define SIMPLE_CLASS_EXPORT
#endif
extern int SIMPLE_CLASS_EXPORT g_Vaule; //全局变量
class SIMPLE_CLASS_EXPORT CSimpleClass
{
public:
CSimpleClass(void);
~CSimpleClass(void);
int GetVale(void)const;
};
#endif
SimpleClass.cpp代码实现如下:
//SimpleClass.cpp
#include "SimpleClass.h"
int g_Vaule = 100;
CSimpleClass::CSimpleClass(void)
:m_iValue(100)
{}
CSimpleClass::~CSimpleClass(void)
{}
int CSimpleClass::GetVale(void)const
{
return g_Vaule;
}
如果应用程序中不直接使用g_Vaule,能顺利编译通过,且调用函数GetVale能正确返回100.
但如果在应用程序中直接使用g_Vaule,编译错误提示如下:
1>main.obj : error LNK2001: unresolvedexternal symbol "int g_Vaule" (?g_Vaule@@3HA)
如果将SimpleClass.h中的宏定义修改为下面值,成功编译:
#ifdef DLL_EXPORTS
#define SIMPLE_CLASS_EXPORT __declspec(dllexport)
#else
#define SIMPLE_CLASS_EXPORT __declspec(dllimport)
#endif
总结如下:对于动态库本身必须使用关键字__declspec(dllexport),对于应用程序,如果不使用动态库导出的变量,不使用关键字__declspec(dllimport)也可以保证动态库的正常使用,但实际使用中,还是建议应用程序使用关键字__declspec(dllimport),具体原因,还是上面MSDN的那段话。
动态库与静态库并存
另外,有时我们的程序需要同时提供动态库和静态库库,且都使用一个头文件,为了解决关键字的使用冲突,建议使用如下的宏定义:
#ifdefined DLL_EXPORTS
#ifdefined INSIDE_DLL
#define SIMPLE_CLASS_EXPORT__declspec(dllexport)
#else
#define SIMPLE_CLASS_EXPORT__declspec(dllimport)
#endif
#else
#define SIMPLE_CLASS_EXPORT
#endif
对于动态库本身,需要定义宏DLL_EXPORTS和INSIDE_DLL 使用动态库的应用程序定义宏DLL_EXPORTS
对于静态库,不需要定义DLL_EXPORTS,当然静态库的应用程序也不需要定义。
如此定义,就可以让动态库和静态库的导出都使用同一份头文件。