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

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

显示调用:程序运行时,在需要使用动态库时手动使用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. 三台三层交换机OSPF多区域划分动态路由实验

    一.实验拓扑 二.实验步骤 1.给主机设置IP,网关:给交换机划分VLAN,给VLAN划分端口,给VLAN设置IP 2.启用OSPF.宣告网段(network 网络地址 反掩码 区域名     其中0 ...

  2. Kafka 入门1

    Kafka 简介 作为一个消息中间件,Kafka 以高扩展性.高吞吐量等特点,在互联网项目中被广泛采用. 不清楚 Kafka 的同学,可以先看看这篇文章: http://blog.csdn.net/s ...

  3. Java连载36-IDE使用

    一.主方法注意 每一个类都可以编写一个主方法,但是一般情况下,一个系统只有一个入口,所以主方法一般写一个 二.Myeclipse的使用 1.在workspace中工作区中有一个文件夹.metadata ...

  4. web.xml引入 xml (tomcat 7.0.52) 以上版本报错

    原文地址:https://blog.csdn.net/sdmxdzb/article/details/47728017?locationNum=11 今天在搞工作流,tomcat7.0.57 总是报错 ...

  5. 【Oracle】rman基于时间点恢复

    rman基于时间点恢复 场景: 由于某研究的误操作,导致财务模块的数据丢失,如何使用rman基于时间点恢复数据. 思路 1.克隆数据库的虚拟机,直接对数据库的数据进行恢复 RMAN> shutd ...

  6. Taxi Cab Scheme POJ - 2060 二分图最小路径覆盖

    Running a taxi station is not all that simple. Apart from the obvious demand for a centralised coord ...

  7. [线段树]区间修改&区间查询问题

    区间修改&区间查询问题 [引言]信息学奥赛中常见有区间操作问题,这种类型的题目一般数据规模极大,无法用简单的模拟通过,因此本篇论文将讨论关于可以实现区间修改和区间查询的一部分算法的优越与否. ...

  8. IDEA设置方法参数列表类型自动提示

    默认情况下,IDEA的提示不够完全,可以通过以下设置,将提示功能打开的更完善. 效果如下面俩图所示

  9. Autoware 笔记 No. 5——基于GNSS的定位

    1. 前言 在之前的笔记No.2 中,我们直接采用ndt_matching的方法实现定位,但需要在打开rviz中,通过2D Pose Estimate指定初始位置.加入GNSS后,可以帮助ndt_ma ...

  10. Laravel应用 -- 脚本任务

    大多数项目在业务发展过程中,都需要修复历史数据和定时任务来完成一些业务逻辑,这部分通常都需要通过脚本来完成,一般的框架爱也都提供这部分的功能,学习并使用是工作中的基本要求. 基本流程 commands ...