__declspec(dllexport)与__declspec(dllimport)

区别

       他们都是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,当然静态库的应用程序也不需要定义。

如此定义,就可以让动态库和静态库的导出都使用同一份头文件。

你可能感兴趣的:(c++)