动态库分为二种,一种隐式链接,另一种显示调用。不论哪种动态库,本质都是运行时动态加载

隐式链接:程序运行时,由编译系统自动加载动态库,然后根据程序的引入表进行重定位,当程序退出时自动卸载动态库

显示调用:程序运行时,在需要使用动态库时手动使用LoadLibrary进行加载,当不需要动态库时使用FreeLibrary进行卸载

动态库创建:

1.新建项目,创建项目和解决方案

DynamicLibrary  这是动态库项目

G:\C++Learn\Library  Library文件夹用于存放库相关文件,包含静态库与动态库工程和解决方案

DynamicLibraryDemo  动态库解决方案名称

2.点击下一步,出现如下界面

选择DLL,选择导出符号,如果需要使用MFC就勾选上,最后点击完成

3.然后VS左边解决方案管理器会显示相关文件

这就是个最简单的动态库,编译生成下

4.文件夹生成了相关文件

解决方案目录:

解决方案目录下,项目文件夹DynamicLibrary,Debug解决方案调试目录下面就存放了编译生成的 输入库(.lib)和动态链接库(.dll)

回顾下静态库生成,一对比就会发现静态库生成时没有.dll文件

5. 动态库比静态库生成的文件稍显复杂些,这里稍微介绍下

首先看下DynamicLibrary.h,这里由于在第二步时勾选了导出符号,所以VS会默认产生一个与工程

同名的.h文件包含示例代码,导出了相关类和函数帮助我们了解动态库使用

注意:如果没有勾选导出符号,是不会生成DynamicLibrary.h文件的,需要自己进行创建.h进行导出

 // 下列 ifdef 块是创建使从 DLL 导出更简单的
// 宏的标准方法。此 DLL 中的所有文件都是用命令行上定义的 DYNAMICLIBRARY_EXPORTS
// 符号编译的。在使用此 DLL 的
// 任何其他项目上不应定义此符号。这样,源文件中包含此文件的任何其他项目都会将
// DYNAMICLIBRARY_API 函数视为是从 DLL 导入的,而此 DLL 则将用此宏定义的
// 符号视为是被导出的。
#ifdef DYNAMICLIBRARY_EXPORTS
#define DYNAMICLIBRARY_API __declspec(dllexport)
#else
#define DYNAMICLIBRARY_API __declspec(dllimport)
#endif // 此类是从 DynamicLibrary.dll 导出的
class DYNAMICLIBRARY_API CDynamicLibrary {
public:
CDynamicLibrary(void);
// TODO: 在此添加您的方法。
}; extern DYNAMICLIBRARY_API int nDynamicLibrary; DYNAMICLIBRARY_API int fnDynamicLibrary(void);

7-11行的宏定义在不同工程中起到不同的作用。

在当前动态库工程中由于定义了DYNAMICLIBRARY_EXPORTS宏, 所以这个宏表示的是导出

而在外部程序,虽然使用动态库包含了这个头文件,但是没有定义这个宏,所以这个宏表示的是导入

13-22行用到的宏也是一样的作用,在当前动态库工程中代表的是导出,在外部程序中代表的导入

可能有人会发现工程下文件并没有定义这个宏,为什么会说这个宏己经被定义呢?

答案就是这个宏在VS的预处理器定义了,创建动态库项目时,VS会自动帮你在预处理器中添加一个与工程项目

名相同,全部大写后面加上_EXPORTS的一个宏

当然也可以在头文件中修改宏的名字,但是相应的预处理器定义里也得保持一致

DynamicLibrary.cpp文件

// DynamicLibrary.cpp : 定义 DLL 应用程序的导出函数。
//


#include "stdafx.h"
#include "DynamicLibrary.h"


// 这是导出变量的一个示例
DYNAMICLIBRARY_API int nDynamicLibrary=0;


// 这是导出函数的一个示例。
DYNAMICLIBRARY_API int fnDynamicLibrary(void)
{
return 42;
}


// 这是已导出类的构造函数。
// 有关类定义的信息,请参阅 DynamicLibrary.h
CDynamicLibrary::CDynamicLibrary()
{
return;
}

 

6. 给CDynamicLibrary类添加一个MathAdd方法,另外添加一个全局函数MathSub. 最后编译生成

DynamicLibrary.h


// 下列 ifdef 块是创建使从 DLL 导出更简单的
// 宏的标准方法。此 DLL 中的所有文件都是用命令行上定义的 DYNAMICLIBRARY_EXPORTS
// 符号编译的。在使用此 DLL 的
// 任何其他项目上不应定义此符号。这样,源文件中包含此文件的任何其他项目都会将
// DYNAMICLIBRARY_API 函数视为是从 DLL 导入的,而此 DLL 则将用此宏定义的
// 符号视为是被导出的。
#ifdef DYNAMICLIBRARY_EXPORTS
#define DYNAMICLIBRARY_API __declspec(dllexport)
#else
#define DYNAMICLIBRARY_API __declspec(dllimport)
#endif


// 此类是从 DynamicLibrary.dll 导出的
class DYNAMICLIBRARY_API CDynamicLibrary {
public:
CDynamicLibrary(void);
int MathAdd(int a,int b);//加法计算(类成员函数)
};


DYNAMICLIBRARY_API int MathSub(int a ,int b);//减法计算(全局函数)

DynamicLibrary.cpp


// DynamicLibrary.cpp : 定义 DLL 应用程序的导出函数。
//


#include "stdafx.h"
#include "DynamicLibrary.h"


//减法计算(全局函数)
DYNAMICLIBRARY_API int MathSub(int a ,int b)
{
return a - b;
}


// 这是已导出类的构造函数。
// 有关类定义的信息,请参阅 DynamicLibrary.h
CDynamicLibrary::CDynamicLibrary()
{
return;
}
//加法计算(类成员函数)
int CDynamicLibrary::MathAdd(int a,int b)
{
return a + b;
}

隐式链接项目创建:

1.给解决方案添加一个新的控制台项目HideLink用于测试动态库,创建完成后设置为启动项目

2.HideLink.cpp添加相关代码


// HideLink.cpp : 定义控制台应用程序的入口点。
//


#include "stdafx.h"
#include "../DynamicLibrary/DynamicLibrary.h" //动态库头文件


#pragma comment(lib,"../Debug/DynamicLibrary.lib")//引入动态库.lib文件
int _tmain(int argc, _TCHAR* argv[])
{
//使用动态库中CDynamicLibrary类,调用其中的MathAdd方法
CDynamicLibrary dynamicLib;
int nResult = dynamicLib.MathAdd(1,2);
printf("1 + 2 = %d\r\n",nResult);
//调用动态库中的全局函数MathSub
nResult = MathSub(5,1);
printf("5 - 1 = %d\r\n",nResult);
getchar();
return 0;
}


3. 编译运行HideLink, 查看相关信息

控制台输出调试信息,下方的模快里不单有HideLink.exe以及系统dll,还多了个动态库DynamicLibrary.dll

总结:虽然动态库的隐式链接方式和静态库调用代码差不多,但是二者内部实现是有明显差别的。

静态库在链接阶段是直接打包到程序中,而动态库的隐式链接在链接阶段只是把用到的函数声明加入

到引用表中,并不会把函数实现代码加入程序中,更不会把整个动态库打包进程序

VC 静态库与动态库(三)动态库创建与使用_隐式链接的更多相关文章

  1. [转]静态库、动态库,dll文件、lib文件,隐式链接、显式链接

    转自:https://blog.csdn.net/dcrmg/article/details/53427181 静态链接.动态链接 静态库和动态库分别应用在静态链接方式和动态链接方式中,所谓静态链接方 ...

  2. 静态库、动态库,dll文件、lib文件,隐式链接、显式链接浅见

    静态链接.动态链接 静态库和动态库分别应用在静态链接方式和动态链接方式中,所谓静态链接方式是指在程序执行之前完成所有的链接工作,把静态库一起打包合入,生成一个可执行的目标文件(EXE文件).所谓动态链 ...

  3. Selenium4+Python3系列(六) - Selenium的三种等待,强制等待、隐式等待、显式等待

    为什么要设置元素等待 直白点说,怕报错,哈哈哈! 肯定有人会说,这也有点太直白了吧. 用一句通俗易懂的话就是:等待元素已被加载完全之后,再去定位该元素,就不会出现定位失败的报错了. 如何避免元素未加载 ...

  4. VC学习笔记---ATL MFC CLR三个库的区别

    MFC.ATL和CLR是VC2005内置的三大库,涵盖了Windows的各种开发方法和开发应用.当然关于C++开发的库不止这三个,不过这三个是微软推荐. 从编程所处层次而言,WIN32为最底层,其次是 ...

  5. Matlab:高阶常微分三种边界条件的特殊解法(隐式Euler)

    函数文件1: function b=F(f,x0,u,h) b(1,1)=x0(1)-h*x0(2)-u(1); b(2,1)=x0(2)+h*x0(1)^2-u(2)-h*f; 函数文件2: fun ...

  6. python+selenium基础篇,三种等待方式,显示、隐式、强制等待

    1.显示等待: from selenium import webdriver from time import sleep from selenium.webdriver.support.ui imp ...

  7. 1.改变项目 动态库、静态库属性。 2.使用运行库 /MD、/MT、/LD说明

    /MD./MT./LD(使用运行库) 有关 C 运行库以及使用 /clr(公共语言运行时编译) 进行编译时要使用哪些库的更多信息,请参见 CRT 库功能. 传递给链接器的给定调用的所有模块都必须使用相 ...

  8. 【1】基于OpenCV的DLL动态库隐式连接

    1DLL的作用 DLL是一个包含可由多个程序同时使用的代码和数据的库.例如:在Windows操作系统中,Comdlg32 DLL执行与对话框有关的常见函数.因此,每个程序都可以使用该DLL中包含的功能 ...

  9. DLL动态库的创建,隐式加载和显式加载

    动态库的创建 打开VS,创建如下控制台工程,工程命名为DllTest: 在弹出的对话框中选择"DLL"后单击"完成"按钮: 在工程中新建DllTest.h和Dl ...

随机推荐

  1. jTopo介绍(一)

    jTopo(Javascript Topology library)是一款完全基于HTML5 Canvas的关系.拓扑图形化界面开发工具包.jTopo关注于数据的图形展示,它是面向开发人员的,需要进行 ...

  2. SDOI 2009 学校食堂

    洛谷 P2157 [SDOI2009]学校食堂 洛谷传送门 JDOJ 1924: [SDOI2009]学校食堂Dining JDOJ传送门 Description 小F 的学校在城市的一个偏僻角落,所 ...

  3. LG3389 「模板」高斯消元法 高斯消元

    问题描述 LG3389 题解 高斯消元,是用来解\(n\)元一次方程组的算法,时间复杂度\(O(n^3)\) 这样就构造出了这个方程组的矩阵 目标就是把这个矩阵左边\(n \times n\)消为单位 ...

  4. 数据仓库009 - SQL命令实战 - where GROUP BY join 部门综合案例

    一.where条件 WHERE 子句中主要的运算符,可以在 WHERE 子句中使用,如下表: 运算符 描述 = 等于 <> 不等于.注释:在 SQL 的一些版本中,该操作符可被写成 != ...

  5. 数据分析常用的Excel函数

    Excel是我们工作中经常使用的一种工具,对于数据分析来说,这也是处理数据最基础的工具. 本文对数据分析需要用到的函数做了分类,并且有详细的例子说明,文章已做了书签处理,点击可跳转至相应位置. 函数分 ...

  6. border和outline的区别

    如果有一个需求,给一个元素增加一条边框,想必大家会习惯且娴熟的使用border来实现,我也是这样   但其实outline也能达到同样的效果,并且在有些场景下会更适用,比如下面的demo 使用bord ...

  7. learning rate warmup实现

    def noam_scheme(global_step, num_warmup_steps, num_train_steps, init_lr, warmup=True): ""& ...

  8. NimSystem实现

    题目 题目比较长,我直接放截图吧 简述 一个比较经典的类与对象的题目,三个类实现了一个比较简单的系统,具体的每个类的要求可以从上面的题目描述中看出(只要你有耐心读完..),不再赘述,代码如下 代码实现 ...

  9. 关于 Scrapy 中自定义 Spider 传递参数问题

    实际应用中,我们有可能在启动 Scrapy 的时候自定义一些参数来控制不同的业务流程,Google 尝试了如下方式可以实现 . 修改 Spider 构造函数  class myspider(Spide ...

  10. keepalived+Nginx实现主备保障Nginx的高可用。

    1.什么是keepalived? Keepalived是集群管理中保证集群高可用的一个服务软件,用来防止单点故障. Keepalived的作用是检测web服务器的状态,如果有一台web服务器死机,或工 ...