1.建立动态库

1.新建win32控制台应用程序, 在 下一步 时选择 "Dll" 动态库

2. Dll.cpp

#include "stdafx.h"

#define DLL_API _declspec(dllexport)
using namespace std;

DLL_API int add(int a,int b){
return a + b;
}

DLL_API int sub(int a,int b){
return a - b;
}

说明:_declspec(dllexport) ,将函数设置为导出函数

动态链接库里 有两种函数类型: 内部 函数 和 导出函数,

内部函数只能在动态库内部访问,导出函数 为 外部访问函数


3.按f7 ,在  .sln 同级目录 debug 下  生成 .lib 和 .dll


2.使用动态库

1.新建 win32控制台程序 ,

2.将  Dll.lib 和 Dll.dll 拷入到 .cpp 同级目录下, 

3.test.cpp

#include "stdafx.h"
#pragma comment(lib,"Dll")
extern int add(int,int);
extern int sub(int,int);

int _tmain(int argc, _TCHAR* argv[])
{
int a = 10;
int b = 20;
printf("%d",add(a,b));
return 0;
}


其中#pramga comment(lib,"动态库名") 是一个预编译指令,用于 加载 库文件

用的比较多的 还有 #pramga once


3.说明,

静态库加载方式需要 :  

1. .h 文件 和 .lib文件 加入到 .cpp同级目录中

2.代码方式:

#include "xxx.h"
#pragma comment(lib,"xxx")

动态库加载方式需要 :  

1..lib 文件 和 .dll 文件

2.代码方式:

#pragma comment(lib,"Dll")

extern int add(int,int);

extern int sub(int,int);



4.关于动态库 和静态库的 区别

1 、 两个lib文件

我们发现,无论是静态链接库还是动态链接库,最后都有lib文件,那么两者区别是什么呢?其实,两个是完全不一样的东西。staticCai.lib的大小为4KB,dllCai.lib的大小为2KB,静态库对应的lib文件叫静态库,动态库对应的lib文件叫导入库。实际上静态库本身就包含了实际执行代码、符号表等等,而对于导入库而言,其实际的执行代码位于动态库中,导入库只包含了地址符号表等,确保程序找到对应函数的一些基本地址信息。

2 、 对于静态链接库,我们在编译和链接的时候已经将所有的代码都导入进来,因此,当生成可执行文件以后,可执行文件包含所有的代码。因此,在可执行文件运行时就不再需要静态库了,这也是为什么我们删掉staticCai.lib程序照样执行;而对于动态链接库,实际上,可执行文件不包含DLL中的内容,只是通过导入库(.lib)知道了相应的地址信息,因此,可执行文件在运行时动态得去加载DLL,这也是为什么我们删掉dllCai.dll后程序就不能执行了。

3 、 对于DLL,我们是可以不要lib文件的。

如果不要lib文件,我们可以通过函数指针的使用达到我们的目的:

#define  DLL_API _declspec(dllexport)
#include <iostream>
using namespace std;   //注意这里的extern "C" , 这里必须加
extern "C" DLL_API int add(int a,int b)   //实现两个整数相加
{
       return a+b;
}
extern "C" DLL_API int subtract(int a,int b)   //实现两个整数相减
{
       return a-b;
}
#include <iostream>
#include <Windows.h>
using namespace std ;
 
typedef int (*func)(int x , int y);  //函数指针
int main()
{
       HINSTANCE hInstance = LoadLibrary("DLL.dll");
       if(hInstance == NULL)
       {
              cout << "SB" << endl ;
              return 0;
       }
       func add = (func)GetProcAddress(hInstance, "add");
       func sub = (func)GetProcAddress(hInstance, "subtract");
       cout << (*add)(3,4) << endl ;
       cout << (*sub)(5,3) << endl ;
}



显然,这种方法没有用lib文件方便,如果为了每次调用一个函数还要自己再弄一个函数指针,多麻烦啊,所以,我们在实际开发中,用的众多的第三方扩展库,别人都是提供的:


.h  文件(类,函数的声明)

.dll 文件(类或函数的实现)

.lib 文件(导入库)



小结:

一、静态库

* 静态库是把程序运行时需要使用的函数编译在一个二进制文件中,扩展名为.lib。当程序link时把静态库中的二进制数据和程序其它数据放到一起。程序运行时不在需要lib和dll文件的支持。这样做的坏处是开发出来的程序占用磁盘空间较大。特别是windows系统中本来就有或很多程序运行都需要的函数完全没有必要每次开发程序时都要使用各自的静态库。

* 静态库为.lib文件形式存在

* 链接后产生的可执行文件包含了所有需要调用的函数的代码,因此占用磁盘空间较大

* 如果有多个(调用相同库函数的)进程在内存中间时运行,内存中就存有多份相同的库函数代码,因此占用内存空间较多。

二、动态库

* 动态库在开发时仅是把dll中的函数名和参数放到应用程序中,应用程序运行时根据函数名和参数调用dll中的函数来运行,这样操作系统中的应用程序可以同时使用同一个dll。可以有效地节省硬盘空间,当然这样做使得程序设计更有层次。也有利于软件工程师的分工和信息安全

* 动态库以.dl文件形式存在,且一般都有一个对应的引入库以.lib文件形式存在。纯资源dll不生成.lib引入库。

   >引入库和静态库的扩展名均为*.lib,但是引入库仅包含一些函数名和参数信息,没有函数体,是为调用动态库服务的,它和动态库的关系相当于.h文件和.cpp文件之间的关系;

* 动态库两种绑定方式

   >静态绑定(static blnding) 使用静态绑定的程序在一开始载入内存的时候,载入程序就会把程序所有调用到的动态代码的地址算出、确定下来。这种方式使程序刚运行时的初始化时间较长,不过一但完成动态装载,程序的运行速度就很快。

    2动态绑定(dynamic binding)   使用这种方式的程序并不在一开始就完成动态链接,而是直到真正调用动态库代码时,载入程序才计算(被调用的那部分)动态代码的逻辑地址,然后等到某个时候,程序又需要调用另外某块动态代码时,载入程序才又去计算这部分代码的逻辑地址。所以,这种方式侄程序初始化时间较短,但运行期间的性能比不上静态绑定的程序。

* 使用动态库的两种方法(windows)

  >方法一: load-time dynamic linking 
在要调用dll的应用程序链接时,将dll的输入库文件(import library,.lib文件)包含进去。具体的做 法是在源文件开头加一句#include ,然后就可以在源文件中调用dlldemo.dll中的输出文件了。

  >方法二: run-time dynamic linking 
不必在链接时包含输入库文件,而是在源程序中使用LoadLibrary或LoadLibraryEx动态的载入dll。