静态链接库LIB和动态链接库DLL的区别 创建和示例

时间:2021-08-08 19:30:43
  1.什么是静态连接库,什么是动态链接库
         静态链接库与动态链接库都是共享代码的方式,如果采用静态链接库,则无论你愿不愿意,lib 中的指令都全部被直接包含在最终生成的 EXE 文件中了。但是若使用 DLL,该 DLL 不必被包含在最终 EXE 文件中,EXE 文件执行时可以“动态”地引用和卸载这个与 EXE 独立的 DLL 文件。静态链接库和动态链接库的另外一个区别在于静态链接库中不能再包含其他的动态链接库或者静态库,而在动态链接库中还可以再包含其他的动态或静态链接 库。静态链接库与静态链接库调用规则总体比较如下。

对于静态链接库(比较简单):
     首先,静态链接库的使用需要库的开发者提供生成库的.h头文件和.lib文件。

          生成库的.h头文件中的声明格式如下:
            extern "C" 函数返回类型 函数名(参数表);
             在调用程序的.cpp源代码文件中如下:
              #include "..\lib.h"
              #pragma comment(lib,"..\\debug\\libTest.lib")

             //指定与静态库一起链接

       第二,因为静态链接库是将全部指令都包含入调用程序生成的EXE文件中。因此如果用的是静态链接库,那么也就不存在“导出某个函数提供给用户使用”的情况,要想用就得全要!要不就都别要!:)

 

对于动态链接库:
     动态链接库的使用,根据不同的调用方法,需要提供不同的资源:

        1. 静态加载,也称隐式加载------程序静态编译的时候就静态导入dll,这样的话就需要提供给库使用者(C客户)如下文件:*.lib文件和.dll文件和*.h。

            其有2个坏处:

    1   程序一开始运行就需要载入整个dll,无法载入程序就不能开始运行;
    2   由于载入的是整个dll,需要耗费资源较多

         其调用方法如下:

               #include "..\lib.h"
               #pragma comment(lib,"..\\debug\\libTest.lib")

              但是这种方式的话可以调用Class method.

      2. 动态加载(运行时加载),也称为 显示加载 --------那么只需要提供dll文件。

            因此调用程序若想调用DLL中的某个函数就要以某种形式或方式指明它到底想调用哪一个函数。但是无法调用Class method了。

     如果要调用Dll中的function,需要经历3个步骤:
Handle h=LoadLibrary(dllName) --> GetProcAddress(h,functionName) 返回函数指针,通过函指针调用其function-->FreeLibrary(h)
          例如:Another.dll有一个int Add(int x,int y)函数。则完整的调用过程如下:
                 typedef int (* FunPtr)(int,int);//定义函数指针
                FunPtr funPtr;
                Handle h=LoadLibrary("Another.dll");
               funPtr=(FunPtr)GetProcAddress(h,"Add");
               funPtr(2,3);//2+3;
              FreeLibrary(h);

 

 

2.  动态链接库的实现

     首先我们必须先注意到DLL内的函数分为两种:
    (1)DLL 导出函数,可供应用程序调用;
    (2)DLL 内部函数,只能在 DLL 程序使用,应用程序无法调用它们。

      动态链接库的export 需要在在相应的头文件中编写相应的MACRO
     自定义了一些类(函数)export 宏(该文件由IDE自动生成)如下
------------------------------------------------------------------
#ifdef MYDLL_EXPORTS
#define MYDLL_API __declspec(dllexport)
#else
#define MYDLL_API __declspec(dllimport)
#endif

这是导出类的宏定义,将导出类必须加上该宏,才能被导出。
此处的MYDLL_EXPORTS会出现在 project-->settings-->C/C++页面上的 PreProcessor definition中,这个MACRO表明其要定义一个导出宏

#ifdef MYDLL_EXPORTS
#define MYDLL_API __declspec(dllexport)
#else
#define MYDLL_API __declspec(dllimport)
#endif

class MYDLL_API CMyClass
{
public:
CString SayHello(CString strName);
CMyClass();
virtual ~CMyClass();

};

 或者直接声明

class CMyClass  
{
public:
__declspec(dllexport) CString SayHello (CString strName);
__declspec(dllexport) CMyClass();
__declspec(dllexport) virtual ~CMyClass();

};