windows 动态链接库

1,dll和lib的关系

lib是静态链接库;DLL是动态链接库,一个编译时提供;一个运行时提供。

其实没那么简单! lib也有静态lib和动态lib之分

静态lib:它将导出声明(后面会讲)和实现均放到lib中,编译后所有代码都嵌入到宿主程序中去。

动态lib:相当于一个h文件,它是对实现部分(.DLL)的导出部分的声明。编译后只是将导出声明部分编译到宿主程序中,运行时需要相应的DLL文件的支持,否则无法工作。当生成一个新的DLL时,也会有配套的lib产生(即二者需一起分发),此时的lib即为动态lib(后面会有还有实验)。


2,创建dll
  1. // mydll.h file  
  2. extern "C" _declspec(dllexportint add(int a, int b);  
  3.   
  4. //mydll.cpp file  
  5. #include "mydll.h"  
  6. int add(int a, int b) //该DLL需要导出的函数功能:加法  
  7. {  
  8.     return a + b;  
  9. }  
说明:

(1)前面的 extern “C” 告诉编译器函数可以在本模块或其他模块中使用,其中“C”表明需按照C语言方式编译和连接它,因为C++编译时,会对函数名进行修饰,用于实现函数重载,而C里面没有这个功能,所以需要用extern "C"在头文件进行声明的时候加以区分,以便链接时能进行正确地函数名查找。

(2)_declspec(dllexport)为导出函数关键字,意为需从DLL中导出该函数,以便使用。

(3)在进行编译连接后会在Debug目录下找到DLL文件和对应的lib文件

3,调用dll

(1)单纯使用.dll

  1. #include <wtypes.h>   
  2. #include <winbase.h>   
  3. #include <iostream> 

  4. _declspec(dllimportint add(int a, int b); //导入声明,亦可以不加,如果加上可加快程序运行  
  5. typedef int(*padd)(int a,int b);  

  6. int main()  
  7. {
  8.     HINSTANCE hDLL;
  9.     hDLL=LoadLibrary("mydll.dll");  //加载 DLL文件  
  10.     if(hDLL == NULL)
  11.         std::cout<<"Error!!!\n";

  12.     padd myadd;
  13.     myadd=(padd)GetProcAddress(hDLL,"add");  //取DLL中的函数地址,以备调用  
  14.     int a = myadd(5,8);   //注意:此时是myadd
  15.     std::cout<<"a: "<<a<<std::endl;
  16.     
  17.     FreeLibrary(hDLL);
  18.     return 0;  


#include <wtypes.h>   
#include <winbase.h>   
#include <iostream>  
_declspec(dllimport) int add(int a, int b); //导入声明,亦可以不加,如果加上可加快程序运行  
_declspec(dllimport) int sub(int a, int b);

typedef int(*padd)(int a,int b);
typedef int(*psub)(int a,int b);

int main()
{
    HINSTANCE hDLL;  
hDLL=LoadLibrary("mydll.dll");  //加载 DLL文件  
    if(hDLL == NULL)
std::cout<<"Error!!!\n"; 

    padd myadd;
    myadd = (padd)GetProcAddress(hDLL,"add");  //取DLL中的函数地址,以备调用
    int a = myadd(8,5);
    std::cout<<"myadd return value = "<<a<<std::endl;

    psub mysub;
    mysub = (psub)GetProcAddress(hDLL,"sub");
    int b = mysub(8,5);
    std::cout<<"mysub return value = "<<b<<std::endl;
    
    FreeLibrary(hDLL);

    system("pause");
    return 0;
}


(2).h + .lib + .dll 结合方式

  1. #include<wtypes.h>   
  2. #include <winbase.h>   
  3. #include <iostream>  
  4. #include "mydll.h"  
  5. #pragma comment(lib,"mydll.lib")  //将mydll.lib库文件连接到目标文件中(即本工程)  
  6. extern "C"_declspec(dllimportint add(int a,int b);  
  7. int main()  
  8.     int a =add(5,8);  
  9.     std::cout<<"a: "<<a<<std::endl;  
  10.   
  11.     return 0;  
  12. }  

#include <wtypes.h>   
#include <winbase.h>   
#include <iostream>

#include "mydll.h"

//将mydll.lib库文件连接到目标文件中
//注:也可以添加到“附加依赖项”
#pragma comment(lib,"mydll.lib") 
extern "C" _declspec(dllimport) int add(int a, int b);  

int main()
{
    int a = add(8,5);
    std::cout<<"myadd return value = "<<a<<std::endl;
    
    system("pause");
    return 0;
}


参考文章:http://blog.csdn.net/heyabo/article/details/8721611

不错文章:http://jingyan.baidu.com/article/f0e83a25ad21b522e4910171.html
不错文章:http://blog.csdn.net/m372897500/article/details/7246368



/

参考文章:http://www.cnblogs.com/TenosDoIt/p/3203137.html

首先介绍一下静态库(静态链接库)、动态库(动态链接库)的概念,首先两者都是代码共享的方式。

静态库在链接步骤中,连接器将从库文件取得所需的代码,复制到生成的可执行文件中,这种库称为静态库,其特点是可执行文件中包含了库代码的一份完整拷贝;缺点就是被多次使用就会有多份冗余拷贝。即静态库中的指令都全部被直接包含在最终生成的 EXE 文件中了。在vs中新建生成静态库的工程,编译生成成功后,只产生一个.lib文件

动态库:动态链接库是一个包含可由多个程序同时使用的代码和数据的库,DLL不是可执行文件动态链接提供了一种方法,使进程可以调用不属于其可执行代码的函数。函数的可执行代码位于一个 DLL 中,该 DLL 包含一个或多个已被编译、链接并与使用它们的进程分开存储的函数。在vs中新建生成动态库的工程,编译成功后,产生一个.lib文件和一个.dll文件

那么上述静态库和动态库中的lib有什么区别呢?

静态库中的lib:该LIB包含函数代码本身(即包括函数的索引,也包括实现),在编译时直接将代码加入程序当中

动态库中的lib:该LIB包含了函数所在的DLL文件和文件中函数位置的信息(索引),函数实现代码由运行时加载在进程空间中的DLL提供

总之,lib是编译时用到的,dll是运行时用到的。如果要完成源代码的编译,只需要lib;如果要使动态链接的程序运行起来,只需要dll

以下例子均在vs2010上测试


生成和使用动态库

生成动态库

 新建项目--win32项目--填写项目名--确定--下一步--应用程序类型:选择dll--附加选项:选择导出符号--完成

可以看到生成了一个dllmain.cpp 文件,这是dll应用程序的入口,注意它和普通工程的入口main函数不同,这个文件我们不需要修改。

 在这个动态库中我们举例导出一个变量,一个类,一个函数,头文件dll.h如下:

复制代码
 1 //新建生成dll的工程时,vs默认定义了宏DLL_EXPORT,因此,DLL_API 是 __declspec(dllexport),用来导出
 2 //当我们在静态调用dll时,我们包含该头文件,由于没有定义DLL_EXPORT,所以DLL_API是
 3 //__declspec(dllimport),用来导入
 4 #ifdef DLL_EXPORTS
 5 #define DLL_API __declspec(dllexport)
 6 #else
 7 #define DLL_API __declspec(dllimport)
 8 #endif
 9 
10 // 导出类
11 class DLL_API Cdll {
12 public:
13     Cdll(void);
14     // TODO: 在此添加您的方法。
15 };
16 
17 //导出变量,变量在.cpp文件中定义
18 extern DLL_API int ndll;
19 
20 //导出函数,加extern "C",是为了保证编译时生成的函数名不变,这样动态调用dll时才能
21 //正确获取函数的地址
22 extern "C" DLL_API int fndll(void);
复制代码

dll.cpp 文件如下:

复制代码
 1 #include "dll.h"
 2 
 3 
 4 // 这是导出变量的一个示例
 5 DLL_API int ndll=6;
 6 
 7 // 这是导出函数的一个示例。
 8 DLL_API int fndll(void)
 9 {
10     return 42;
11 }
12 
13 // 这是已导出类的构造函数。
14 // 有关类定义的信息,请参阅 dll.h
15 Cdll::Cdll()
16 {
17     return;
18 }
复制代码

调用动态库

有两种方法调用动态库,一种隐式链接,一种显示链接。

调用动态库:隐式链接

隐式链接 需要.h文件,dll文件,lib文件

(1)将dll放到工程的工作目录

(2)设置项目属性--vc++目录--库目录为lib所在的路径

(3)将lib添加到项目属性--链接器--输入--附加依赖项(或者直接在源代码中加入#pragma comment(lib, “**.lib”)

(4)在源文件中添加.h头文件

然后就像平常一样调用普通函数、类、变量

调用动态库:显示链接

显示链接 只需要.dll文件,但是这种调用方式不能调用dll中的变量或者类(其实可以调用类,但是相当麻烦,有兴趣者可参考http://blog.csdn.net/jdcb2001/article/details/1394883

显示调用主要使用WIN32 API函数LoadLibrary、GetProcAddress,举例如下:

复制代码
 1 typedef int (*dllfun)(void);//定义指向dll中函数的函数指针
 2     HINSTANCE hlib = LoadLibrary(".\\dll.dll");
 3     if(!hlib)
 4     {
 5         std::cout<<"load dll error\n";
 6         return -1;
 7     }
 8     dllfun fun = (dllfun)GetProcAddress(hlib,"fndll");
 9     if(!fun)
10     {
11         std::cout<<"load fun error\n";
12         return -1;
13     }
14     fun();
复制代码

生成和使用静态库

生成静态库

新建项目--win32项目--填写项目名--确定--下一步--应用程序类型:选择静态库

静态库项目没有main函数,也没有像dll项目中的dllmain。

创建项目后添加.h文件,添加相应的导出函数、变量或类,如下所示

复制代码
 1 #ifndef _MYLIB_H_
 2 #define _MYLIB_H_
 3 
 4 void fun(int a);
 5 
 6 extern int k;
 7 
 8 class testclass
 9 {
10 public:
11     testclass();
12     void print();
13 };
14 
15 #endif
复制代码

.cpp文件如下

复制代码
 1 #include "stdafx.h"
 2 #include "lib.h"
 3 #include <iostream>
 4 
 5 void fun(int a)
 6 {
 7     std::cout<<a<<"lib gen\n";
 8 }
 9 
10 int k = 222;
11 
12 testclass::testclass()
13 {
14     std::cout<<"123\n";
15 }
16 
17 void testclass::print()
18 {
19     std::cout<<"this is testcalss\n";
20 }
复制代码

编译工程后就会生成一个.lib文件

使用静态库

需要.h文件,lib文件

(1)设置项目属性--vc++目录--库目录为lib所在的路径

(2)将lib添加到项目属性--链接器--输入--附加依赖项(或者直接在源代码中加入#pragma comment(lib, “**.lib”))

(3)在源文件中添加.h头文件

然后就像平常一样调用普通函数、类、变量,举例如下:

复制代码
 1 #include <iostream>
 2 #include "lib.h"
 3 
 4 #pragma comment(lib, "lib.lib")
 5 
 6 int main()
 7 {
 8     fun(4);
 9     std::cout<<k<<std::endl;
10     testclass tc; 
11     tc.print();
12     return 0;
13 }
复制代码


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值