链接库动态链接库详细介绍

windows中,链接库分为两种类型:静态链接库.lib和动态链接库.dll。其中动态链接库在被使用的时候,通常还提供一个.lib,称为引入库,它主要提供被Dll导出的函数和符号名称,使得链接的时候能够找到dll中对应的函数映射。
静态链接库和动态链接库的作用相似,都是提供给其他程序进行调用的资源。其中,动态链接库的调用方法分隐式调用(静态导入调用)和显示调用(动态导入调用)。

编译环境
Microsoft Visual Stdio 2010
--------------------------------------------------------------------------------
DLL导出符号
例,首先生成一个dll1.dll和dll1.lib


代码如下:

// DLL1工程,dll1.cpp
// _declspec(dllexport)为导出符号
_declspec(dllexport) int add(int a, int b)
{
return a + b;
}

利用微软的depends工具查看dll1.dll,导出的符号如下:
 
其中各字段意义:Ordinal(符号序号,后面使用GetProcAddress的时候,参考的数值),Hint(这个我也不是太明白,据说是不用了解),Function(这个就是函数导出后的符号名称了),EntryPoint(这个是函数在DLL中的地址)。
这里之所以函数的名称变成了这样子,是因为使用的编译器默认使用C++方式进行编译,由于C++支持重载,那么需要给函数名增加额外的符号,来使与同名的重载函数区分开来,才能在DLL中通过符号名来进行定位。
这里可以做个简单的测试,新建控制台测试工程DllTest如下。


代码如下:

// DllTest工程,DllTest.cpp
#include <iostream>
using namespace std;
int main(void)
{
// extern int add(int a, int b);
// _declspec(dllimport)是导入声明,这种方式比上面的方式更有效,同时编译器能边编译出更加高效的代码。
_declspec(dllimport) int add(int a, int b);
cout << add(1,2) << endl;
getchar();
return 0;
}

编译链接,提示链接错误 error LNK2019: unresolved external symbol "__declspec(dllimport) int cdecl add(int,int)" (__imp_?add@@YAHHH@Z) referenced in function _main,很明显的编译器在编译的时候,把add函数也给重命名了,并且和上面用depends查看的一样。意思是没有找到这个符号的定义。

添加代码后如下:(注意,我这里两个工程的输出目录都是在和解决方案同目录的debug下,为了避免每次修改都重新拷贝lib文件,直接使用相对路径声明。)


代码如下:

// DllTest工程,DllTest.cpp
#include <iostream>
using namespace std;
#pragma comment(lib, "../debug/dll1.lib") // 显示的声明要链接dll1.lib,隐式调用
int main(void)
{
// extern int add(int a, int b);
// _declspec(dllimport)是导入声明,这种方式比上面的方式更有效,同时编译器能边编译出更加高效的代码。
_declspec(dllimport) int add(int a, int b);
cout << add(1,2) << endl;
getchar();
return 0;
}

编译运行后,使用depends工具对DllTest.exe查看其依赖的输入信息如下:
 
可以看出,DllTest.exe通过dll1.lib,引入了对dll1.dll的依赖。

--------------------------------------------------------------------------------

DLL提供的头文件
通常情况下,当得到一个.dll的时候,我们无法得知其提供了哪些函数调用(准确来说,应该是调用方式。因为我们可以利用depends工具查看dll导出的函数及其序号,当然也许可能有其他的方式去知道具体怎么使用,但是肯定无法得知内部具体实现细节。),因此为了方便被使用,通常会提供一个对应该dll的.h文件,来声明其提供给客户端使用的方式和说明等信息。客户端使用该头文件对所使用的接口进行导入。但是为了避免很多地方都出现这些函数的声明,通常在客户端直接在.h文件中对所有接口进行导入,而在Dll编译时,则作为导出使用。方法如下:


代码如下:

// DLL1工程,dll1.h
#ifndef DLL1_API
#define DLL1_API _declspec(dllimport)
#endif
// 以上代码表示,如果在包含该头文件之前,没有定义DLL1_API宏,那么后面所有DLL1_API宏都展开为_declspec(dllimport),即导入。
// 因为通常情况下客户端不会去定义这个宏(当然,假设这个宏不会被客户端中其他文件定义),所以客户端使用该头文件的时候,都是用于导入。
DLL1_API int add(int a, int b);

代码如下:

// DLL1工程,dll1.cpp
#define DLL1_API _declspec(dllexport)
// 注意上面这行,在头文件被包含前,先定义了DLL1_API这个宏,使得头文件中DLL1_API都被展开为_declspec(dllexport)了,从而声明函数作为导出。
#include "dll1.h"
// 在头文件中进行了导出声明的函数,就不用再声明导出了。
int add(int a, int b)
{
return a + b;
}

相应的,TestDll工程中包含.h文件后,也不用再去申明了。


代码如下:

// DllTest工程,DllTest.cpp
#include <iostream>
using namespace std;
#include "../dll1/dll1.h" // 包含该头文件之后,后面就不需要再申明了
#pragma comment(lib, "../debug/dll1.lib") // 显示的声明要链接dll1.lib,隐式调用
int main(void)
{
cout << add(1,2) << endl;
getchar();
return 0;
}

以上基本解释了为什么通常引用dll的时候都有一个头文件,并且头文件内有很多#ifndef之类的东东了。

--------------------------------------------------------------------------------
动态链接库导出类
当然,动态链接库也能导出类,要注意的是声明的方式为class DLL1_API CSample,而不是DLL1_API class CSample。
同时,要注意导出类的同时,其所有成员函数也已经导出,但是仍然遵循类成员变量访问权限限制。
如果单独导出类的成员函数(声明方式和全局函数一样),那么在客户端可以实例化类对象,并调用导出的成员函数,不能调用没导出的成员函数(即使是public的)。

--------------------------------------------------------------------------------
改编了的符号名
在导出符号时,讲过C++会对函数名进行改编,以支持函数重载。那么就会存在一个问题,如果使用不用的C++编译器(导致编译出的符号名不同)或者客户端使用C编译器调用,就会出现LNK2019这样的链接错误,找不到符号。这个问题很大的限制了DLL的使用范围。
解决方法1:
使用extern “C”(注意这个C一定要大写)前置申明,表明函数是以C的方式编译链接的。C方式编译连接导出的函数不会改编符号名,因而可以避免上述问题。


代码如下:

// DLL1工程,dll1.h
#ifndef DLL1_API
#define DLL1_API extern "C" _declspec(dllimport)
#endif
// 以上代码表示,如果在包含该头文件之前,没有定义DLL1_API宏,那么后面所有DLL1_API宏都展开为_declspec(dllimport),即导入。
// 因为通常情况下客户端不会去定义这个宏(当然,假设这个宏不会被客户端中其他文件定义),所以客户端使用该头文件的时候,都是用于导入。
DLL1_API int add(int a, int b);
//class CSample
//{
//public:
// DLL1_API int substract(int a,int b);// 这种情况下,导出类成员函数,编译不能通过的
//};

这里要注意.h和.cpp中都要加上extern “C”,使得导入和导出都用C编译方式。


代码如下:

// DLL1工程,dll1.cpp
#define DLL1_API extern "C" _declspec(dllexport)
// 注意上面这行,在头文件被包含前,先定义了DLL1_API这个宏,使得头文件中DLL1_API都被展开为_declspec(dllexport)了,从而声明函数作为导出。
#include "dll1.h"
// 在头文件中进行了导出声明的函数,就不用再声明导出了。
int add(int a, int b)
{
return a + b;
}
//
//int CSample::substract(int a,int b)
//{
// return a - b;
//}

然后用depends进行查看:
 
导出的函数符号名和函数声明时一样了。由于客户端使用的时候,导入也是用的extern “C”方式,因此客户端在编译链接的时候,也使用的是函数原名称符号。
显示,由于使用的是C编译链接方式,C++的类和成员函数的导出就不能用这种方式了。
此外,如果我们在给函数声明加上标准调用约定:DLL1_API int _stdcall add(int a, int b);(注意,在函数的定义中也要加上_stdcall)。那么编译出来的结果使用depends查看符号名为_add@8,也就是说符号名称又改了。
解决方法2:
使用模块定义文件.def,这种文件的格式规范查看MSDN,搜索.def即可。其中LIBRARY命令用于指名该def文件用于导出库文件,EXPORTS用于指名导出函数符号名。也就是说,.def文件主要就用于控制导出符号等信息。


代码如下:

LIBRARY dll1
EXPORTS
add11=add

我这里给add函数别名为add11,注意同时需要在.h文件中声明add为add11(作用就是提供给客户端使用,当然其实也可以直接在客户端声明函数为add11,前提是你知道该函数的定义方式等)。一旦提供了.def之后,.cpp中提供的任何调用约定都不再生效,因为.def指定了生成的符号名了。这里只要明白,.def控制了Dll的导出符号,客户端使用的时候,只要提供了声明,并且链接上.lib文件,就能够使用了。
补充一句:VC6.0以后的IDE都需要在链接选项(LINK)的input-》module define file中,指明.def文件,编译器才会去使用这个.def文件。

备注:
这一块关于调用约定,以及加不加extern “C”,目前我还比较混乱。等了解了之后,再来这里进行补充,欢迎看客给我提供有关这一块比较好的介绍资料。

--------------------------------------------------------------------------------
显示链接(动态导入链接)
前面有提到让导出的函数没有名字,那么客户端如何对其进行调用呢。就是使用符号表中的ordinal了(可以通过工具进行查看),当然也可以使用函数名进行导入。动态导入不需要lib文件和.h文件(如果知道函数名的话)。


代码如下:

// DllTest工程,DllTest.cpp
#include <iostream>
using namespace std;
#include <windows.h>
int main(void)
{
HMODULE hModule = ::LoadLibraryA("dll1.dll");
if(NULL != hModule)
{
typedef int (*ADDPROC)(int a, int b);
//ADDPROC add = (ADDPROC)::GetProcAddress(hModule, "add11"); // 通过函数名导入
ADDPROC add = (ADDPROC)::GetProcAddress(hModule, MAKEINTRESOURCEA(1)); // 通过ordinal导入,注意第二个参数此时数值要放在低字节位置,具体参见MSDN说明。
if(NULL != add)
cout << add(1,1) << endl;
::FreeLibrary(hModule);
}
getchar();
return 0;
}

值得一提的是,根据函数名称进行动态导入的时候,实际上也是根据符号名的,也就是说,dll提供的符号名要和提供给用户的函数声明一致。

此外,动态导入动态链接库时,.exe时无法查看到需要的输入信息的。

--------------------------------------------------------------------------------
后记
第一次写这么详细和这么长的博文,可以看出来写到后面的时候已经越来越粗糙了。因为写到后来的时候,自己都已经不知道要写什么了,因为如果再扩展下去,需要增加的东西就太多了。同时也有一部分原因是后面的内容我理解的不是太透彻,同时使用的也很少。总之这会我脑子已经相当混乱了。在这里对那些写博客的大侠们表示深深的敬佩。

最后要申明的是,这些都是我自己参考视频整理总结出来的,肯定有不专业甚至不对的地方,恳请提出指导,我在学习理解之后,会及时进行修改,谢谢。

(0)

相关推荐

  • Python调用C/C++动态链接库的方法详解

    本文以实例讲解了Python调用C/C++ DLL动态链接库的方法,具体示例如下: 示例一: 首先,在创建一个DLL工程(本例创建环境为VS 2005),头文件: //hello.h #ifdef EXPORT_HELLO_DLL #define HELLO_API __declspec(dllexport) #else #define HELLO_API __declspec(dllimport) #endif extern "C" { HELLO_API int IntAdd(in

  • 浅析C/C++中动态链接库的创建和调用

    DLL 有助于共享数据和资源.多个应用程序可同时访问内存中单个DLL 副本的内容.DLL 是一个包含可由多个程序同时使用的代码和数据的库.下面为你介绍C/C++中动态链接库的创建和调用. 动态连接库的创建步骤: 创建Dll有两种方式. 一.创建Non-MFC DLL动态链接库 1.打开File -> New -> Project选项,选择Win32 Dynamic-Link Library ->sample project ->工程名:DllDemo 2.新建一个.h文件DllDe

  • VC6.0如何创建以及调用动态链接库实例详解

    小弟在公司的职责,在上篇博客中已经简约介绍.这边博客主要介绍技术的应用而不在细究原理.因为公司项目着急,出结果要紧,并且咱也不是专注搞研究的,所以,基本懂了原理后,直接上手工作,搞出demo来最好. 至于公司工作情况,今天暂且略过,当然也不是一两句能够表达清楚的.后面会有相应的工作总结,敬请期待-- 现在,废话少说,直奔主题--VC6.0中创建动态链接库. 作为客户与后台的中介,为了更好的调节两方的关系,我明智滴选择了webservice以及动态链接库.在与客户c++使动态链接库方式,而与后台j

  • GCC 编译使用动态链接库和静态链接库的方法

    1 库的分类 根据链接时期的不同,库又有静态库和动态库之分. 静态库是在链接阶段被链接的(好像是废话,但事实就是这样),所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. 有别于静态库,动态库的链接是在程序执行的时候被链接的.所以,即使程序编译完,库仍须保留在系统上,以供程序运行时调用.(TODO:链接动态库时链接阶段到底做了什么) 2 静态库和动态库的比较 链接静态库其实从某种意义上来说也是一种粘贴复制,只不过它操作的对象是目标代码而不是源码而已.因为静态库被链接后库

  • 在C语言中调用C++做的动态链接库

    今天在做东西的时候遇到一个问题,就是如何在C语言中调用C++做的动态链接库so文件 如果你有一个c++做的动态链接库.so文件,而你只有一些相关类的声明, 那么你如何用c调用呢,别着急,本文通过一个小小的例子,让你能够很爽的搞定. 链接库头文件: head.h class A { public: A(); virtual ~A(); int gt(); int pt(); private: int s; }; firstso.cpp #include <iostream> #include &

  • 链接库动态链接库详细介绍

    windows中,链接库分为两种类型:静态链接库.lib和动态链接库.dll.其中动态链接库在被使用的时候,通常还提供一个.lib,称为引入库,它主要提供被Dll导出的函数和符号名称,使得链接的时候能够找到dll中对应的函数映射. 静态链接库和动态链接库的作用相似,都是提供给其他程序进行调用的资源.其中,动态链接库的调用方法分隐式调用(静态导入调用)和显示调用(动态导入调用). 编译环境: Microsoft Visual Stdio 2010 -------------------------

  • c++中STL库队列详细介绍

    1.queue单向队列(先进先出,只能从尾端加元素,从头删元素)         使用方式:在前面加上文件名'#include<queue>',再进行声明'queue<int>m;''其中'<>'里面是数组的类型,'m'是数组的名字.         操作: 1.q.push()//入队 2.q.pop()//让队首出队 3.q.front()//获得队首元素 4.q.back()//获得队尾元素         5.q.empty() 队列是否为空 6.q.size(

  • Vue状态管理库Pinia详细介绍

    目录 什么是 Pinia 如何使用 Pinia 认识 Store 定义一个store 使用 store 操作 State Getters 1. 认识和定义 Getters 2. 访问 Getters 认识和定义 Action 什么是 Pinia Pinia (西班牙语中的菠萝),本质上依然是一个状态管理的库,用于跨组件.页面进行状态共享. pinia 与 vuex 的区别: 更友好的TypeScript支持,Vuex之前对TS的支持很不友好 与 Vuex 相比,Pinia 提供了一个更简单的 A

  • Linux 硬链接和软链接详细介绍

    Linux中的硬链接和软链接 软链接和硬链接 命令ln ln是创建链接的命令: 创建硬链接:ln file link 创建软链接:ln -s file link 硬链接 硬链接(hard link)是Unix系统最早的创建链接的方式. 默认情况下每个文件都有一个硬链接,创建硬链接时,实际上是创建了附加的入口,当且仅当指向文件的所有硬链接都被删除之后文件才被真正删除,即数据块被清理. 为一个文件创建硬链接,类似于拷贝一个文件,但是这个新的拷贝和原先的版本是同步更新的. 这是因为Linux文件系统中

  • 分享十款最出色的PHP安全开发库中文详细介绍

    1. PHP入侵检测系统 PHP IDS(即PHP-入侵检测系统)是一套易于使用.结构良好.速度出色且专门面向PHP类Web应用程序的先进安全层.这套入侵检测系统既不提供任何缓和及杀毒机制,也不会对恶意输入内容进行过滤,其作用单纯为识别出攻击者们针对站点进行的恶意活动.并以大家需要的方式作出及时提醒.凭借着一整套经过实践检验及相当严格的过滤规则,该检测系统会针对任何攻击活动给出一个影响评级数值,从而帮助用户更轻松地了解应如何应对当前出现的黑客攻击.具体应对方式多种多样,包括简单将日志纪录通过紧急

  • IOS 打包静态库详细介绍

    IOS 打包静态库详细介绍 一.前言 前段时间看的一本书上说:"隔着一段距离看,很多有趣的知识看起来都很唬人."比如说这篇我要总结的"静态库知识",在我初出茅庐的时候着实觉得那些后缀名为".frameworke".".a".".dylib"的文件很神秘,很高冷.那时我虽然知道只要导入一个库就能引用库里面很多封装好的东西,但对这个"库"究竟是什么"鬼",一直都是云里雾里

  • 各种Python库安装包下载地址与安装过程详细介绍(Windows版)

    在用Python开发时(Windows环境),会碰到需要安装某个版本的第三方库,为了以后查找.安装方便,总结如下: windows版的各种Python库安装包下载地址: http://www.lfd.uci.edu/~gohlke/pythonlibs/ 一.打开cmd 二.将cmd当前目录切换到Downloads,命令为cd Downloads 三.在文章开头链接地址下载所需第三方库,以SciPy为例:0.18.1是库版本号:cp27.cp35分别对应Python2.7.Python3.5:w

  • C++封装静态链接库和使用的详细步骤

    目录 零碎记事 为什么要把程序封装成库 博主的环境 封装步骤 准备好待封装的程序 开始封装 配置项目 编译 找到编译好的静态库 打包 使用静态库使用步骤包含头文件 添加链接路径 源文件设置 项目设置 零碎记事 距离上次发博客已经有一年半了,转眼间我也是从做图像研究到了做游戏开发,说起来看看前面的博文,本来就有前兆的东西呢(笑)......因为主要还是在使用虚幻引擎,所以C++的东西会碰到多一些. 以后程序技术方面的文章就放博客,游戏设计相关的杂谈就放知乎那边吧,博主的知乎可以通过友链过去. B站

  • IOS 静态库打包流程简化详细介绍

    IOS 静态库打包流程简化 在iOS开发中,我们经常会遇到开发SDK的需求.开发好的静态库后需要手动的合并.a文件,然后再拷贝相关的头文件,接着把静态库和头文件放在同一个文件里面打包发送给SDK的使用者.本文将介绍如何使用脚本,简化这一连串的过程.为了照顾广大初学者,教程将会详细介绍打包的基本流程. 关于静态库和动态库区别,可以看我的另外一篇文章: iOS静态库和Framework区别 项目配置 新建一个名为TestSDK的静态库工程 然后点击Target下边的加号按钮,添加新的Target 选

随机推荐