VisualStudio2019构建C/C++静态库和动态库dll的问题 附源码

1. 静态库和动态库

1.1. 静态链接库

举个例子,假如你在编写一个C++工程,根据业务逻辑,这个工程需要用到一些工具类,例如集合操作的工具类(暂且叫他collection_utils),于是你直接定义一个collection_utils.h头文件和一个collection_utils.cpp文件,在头文件中写一些工具函数的定义,在cpp文件中写函数的实现逻辑;如下所示:

//---------------collection_utils.h-----------------------------
#ifndef COLLECTION_UTILS
#define COLLECTION_UTILS
//合并两个集合
Collection mergeCollection(const Collection& c1,const Collection& c2);
#endif
//---------------collection_utils.h-----------------------------

//---------------collection_utils.cpp-----------------------------
#include "collection_utils.h"
#include <vector>
// .....
Collection mergeCollection(const Collection& c1,const Collection& c2){
	//....实现逻辑
}
//---------------collection_utils.cpp----------------

然后你发现这个工具类具有通用性,在其他项目中也有类似的工具类的需求,想让同事也用上你这个工具类,防止重复造轮子,然后你就把这两个文件发给你的同事,此时聪明的你想起来这样做有个不好的地方,因为项目编译的时候,make工具会逐个编译每个文件生成obj模块,然后通过连接器,把各个模块连接起来,然后打包生成一个exe可执行镜像,这样只要把这个工具类引入任何一个项目,它都要经历编译到obj的过程,但是对于工具类代码来说,几乎是写好了以后就不怎么变化的东西了,这样每个工程都编译一遍,岂不是浪费了时间?而且随着工具类库的增加,这种方法的弊端就会越明显。
那有没有一种方法,可以让这些工具类库代码只编译一次,让连接器在连接的时候,把已经编译好的函数直接拷贝过来,缩短项目的构建时间呢? 答案是肯定的,它就是静态链接库。
有了静态链接库,其他工程只需要在工程中引入函数声明的头文件,在连接的时候,把静态链接库的库文件提供出来就可以完成工程的构建。其实静态库很常见,例如我们用的C标准库中的math.h,如果你包含math.hstdio.h等头文件,这些头文件声明的函数实现不是每次构建工程都会把这里的代码编译一遍的,他们都是以预编译的静态链接库的形式提供,在连接的时候,把我们调用的函数代码指令,从这些库中拷贝到最终的可执行文件中。

1.1. 动态链接库

我们上面说到的静态连接库是把预编译的模块拷贝到自己的模块中,然后打包构建exe镜像,这当然节省了编译器的时间,但是从某种程度上讲,还是有些不足,因为:

  1. 在每一个构建出的每一个exe镜像中,都会有同一个函数的代码拷贝,造成额外的空间开销;
  2. 当这些静态库升级时,所有的模块都要重新编译;

那有没有一种依赖方式,可以让程序在编译时,仅仅记录调用函数的名称,函数的实现代码放在专门的一个地方,这样的库在内存中只装在一份;等到调用时,根据调用函数的名称到库中查找得到函数的入口地址呢?当然有的,那就是动态链接库(dll),顾名思义,这种类型的库是在程序运行时,需要哪个函数,就加载对应的dll到内存中,然后动态把函数调用的符号引用连接到实际的调用地址,当然这一步是由操作系统完成的啦,自己的程序不需要操心,这个比静态库要节省空间,但是会存在动态连接(把符号引用转为直接引用)的过程,对于调用性能要求较高的函数,可能会损失性能。

一般在windows系统中,动态链接库的文件扩展名是.dll,静态链接库的名称是.lib,在linux系统中,动态库的扩展名是.so,静态库的扩展名是.a

2. 使用VisualStudio构建演示

VisualStudio 2019版本:16.8.3(社区版)

2.1. 静态库构建演示

创建一个名称为StaticDynamicLibraryStudy空白解决方案



添加一个静态库项目

项目类型选择静态库

填入名称:StaticLibrary,

最终新建好的项目目录结构如下:

我们可以把pch.cppStaticLibrary.cpp文件删掉,添加自己的代码,举例如下:

添加一个头文件,例如sayHello.h

然后在源文件中新建一个源文件sayHello.cpp,实现sayHello逻辑,如下:

然后,生成项目,在项目上右键,生成:

然后报错了,😂如下:

如果遇到此报错,只需要在项目上右键—>属性,

然后再次生成就可以了,

当然这个目录是可以改的,项目—>右键—>属性—>配置属性—>常规—>输出目录,大家可以去改。
然后在解决方案中增加一个测试控制台项目,名称叫做StaticLibraryTest,新建项目的过程上面有的,不再赘述。删除掉多余的注释,最终得到的项目结构:

因为C++中函数遵守先声明后使用的原则,为了能在新的项目中使用sayHello函数,首先需要声明,因为演示只有这么一个函数,所以你可以在main函数之前,直接声明,

如果需要使用的函数比较多,也可以直接把头文件复制到当前项目,然后include之,我觉得后一种比较规范,我就采用包含头文件的方式了:

目前我们只是解决了声明函数的问题,但是函数的实现代码我们还没有包含进来,函数的实现代码在上一步我们生成的StaticLibrary.lib中,如何包含呢?使用#pragma comment预处理指令,如下所示:

生成项目,然后运行试试,

如何设置当前解决方案运行那个项目的可执行文件呢?解决方案上—>右键—> 属性—>通用属性—>启动项目—>单启动项目,VS设置太多,自己慢慢摸索吧。

然后就会看到如下输出:

说明你成功了。nice~
其实,#pragma comment还可以指定相对路径,是相对连接器构建时的工作目录,在VS里,连接器的工作路径就是项目根路径,例如,改成如下形式,也是可以编译运行的。

当我们需要引入的静态库很多时,都使用绝对路径或相对路径写难免麻烦,我们可以告诉连接器去哪个目录下找库文件,然后只需要在预处理指令中放入我们的静态库的名称即可。VS中提供这种支持,配置方法:项目—>右键—>属性—

>配置属性—>链接器—>常规—>附加库目录

然后把程序改成这样,也可以运行的。当然你把lib文件复制到项目根目录下,不用添加附加目录,直接在预处理指令上写库名称也是可以的。

如果我们这一句也不想写,可以直接在VS中指定包含哪个库,操作方法,项目—>右键—>属性—>配置属性—>链接器—>输入—>附加依赖项

添加我们的库名称,这个时候直接写库的名称,前提是已经配置了附加目录,如果没有配置附加目录,这里需要写全路径或相对路径

然后把程序改成这样,

也是可以运行成功的。

2.2. 动态库构建演示

还是在当前的解决方案里,新建一个项目,项目类型选择动态库,名称是DynamicLibrary

新建以后是这样的:

这里的dllMain是dll的入口点,然后我们在添加sayHello.hsayhello.cpp,只不过头文件需要加上__declspec

(dllexport),如下图:

这个标识的意思是,当前的sayHello函数需要从dll导出,相当于暴漏给外部的服务接口。在cpp文件中我们打印:Hello,I am from dynamic library,然后项目—>右键—>生成,会生成3个文件:

其中lib文件是动态库的导入库文件,这个文件是让连接器在连接的时候,只需要记录调用函数的名称和在dll中的偏移地址,而不去拷贝其代码实现,等到运行的时候,会由操作系统把动态库的地址映射到当前进程的地址空间。
我们现在再添加一个控制台项目DynamicLibraryTest,在里面进行sayHello函数的声明,注意声明时,要用如下方式:

然后还需要像静态库一样,使用#progma commen预处理指令,把lib导入库文件引入进来,具体引入的方法我就不再赘述了,上面有说。最终就像这样:

然后,工程—>右键—>生成,然后运行,结果如下:(这里需要保证你的可执行文件和dll在同一目录,当然把dll文件添加到path路径也是可以的)

这种方式叫做隐式链接,调用函数时,程序是如何找到dll中的入口地址的,完全是连接器帮我们做了,那我们能不能手动找到呢?即在程序运行时,动态的获取到某个函数的句柄? 如果我们只有一个dll文件,没有导入库,但是我们知道里里面的函数声明,这个时候我们该怎么调用呢?下面我们就看看显式链接
要显式链接,首先需要修改一下原来的动态库,VS中新建一个模块定义文件,项目—>右键—>添加—>新建项—>Visual C++ —>代码—>模块定义文件(def)


名称我就叫做DynamicLibrary.def,内容如下:

然后,重新生成,在DynamicLibraryTest项目的main函数中,写上如下代码:

然后,重新生成,运行,有点像Java的反射,结果图我就不贴了。LoadLibrary中的路径可以只使用dll的名称,前提是dll必须在可执行文件同级目录或在path路径中。

3. 总结

以上就是静态库和动态库的所有内容了,本文只是在Windows平台进行演示,后续有空会增加在Linux平台的演示,一步一步教会你,源码已上传Gitee码云仓库,编辑仓促,如有发现错误,请大家不吝赐教。

到此这篇关于VisualStudio2019构建C/C++静态库和动态库dll的问题 附源码的文章就介绍到这了,更多相关VisualStudio2019 dll动态库内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • visual studio 2019编译c++17的方法

    右键点击你的项目打开 属性 > C/C++ > Language > C++ Language Standard,选择c++ 17 或者 c++ latest即可. 步骤1:右键点击项目如下图: 步骤二,选择最下面的"属性"项,如下图: 步骤三,点击"配置属性",选择"通用",然后在右侧选择"C++语言标准",后面的下拉列表中选择自己需要的C++版本编译器即可. 到此这篇关于visual studio 201

  • Visual Studio 2019创建C++ Hello World项目的方法

    最近准备入坑c++ 使用visual studio来配c++环境,虽然visual studio很笨重,但轻松啊~~,安装后什么都不用管,就能跑c++代码了:P 不过启动项目就有一个很尴尬的问题,记录简单一下. 1.新建一个空项目 由于visual studio写代码,一定要新建工程,我们这边选择空的项目 即可 2.新建源文件 注意:只要这样创,源文件才进该项目,才能启动.调试 3.编写代码 4.启动调试 快捷键Ctrl + F5 进行编译和运行,或者点击箭头图标 到此这篇关于Visual St

  • visual studio 2019安装配置可编写c/c++语言的IDE环境

    IDE的下载和安装: 首先,到visual studio官网下载vs2019的安装程序. 学生.或个人开发者免费下载第一个community版本. 下载完成后双击进行安装.安装时要选择安装工作负载.可根据需要勾选相应的负载.因为我们要写c/c++程序,所以勾选了这个使用c++的桌面开发,然后点击右下角安装按钮继续安装. 如果安装的时候忘记选择工作负载,或者安装完成后需要更改工作负载,可以在此打开vs2019的安装程序,然后在点击已安装程序中的修改按钮进行修改: 进入工作负载配置界面做好勾选后点击

  • 如何使用visual studio2019创建简单的MFC窗口(使用C++)

    本文介绍了如何使用visual studio2019创建简单的MFC窗口(使用C++) ```cpp 使用visual studio 2019 创建过程请参考Bili的上一篇文章⬇⬇ →!使用visual studio 2019 创建简单的MFC窗口「使用底层的C语言」 #include<windows.h> //底层实现窗口的头文件 //6.处理窗口过程 //CALLBACK 代表_stdcall 参数的传递顺序:从右到左依次入栈,并且函数返回前清空堆栈 LRESULT CALLBACK W

  • C++运算符重载实例代码详解(调试环境 Visual Studio 2019)

    最近看了菜鸟教程里的C++教程 遇到很多运算符重载,为了方便我的学习,我把这些总结了一下 如有错误(包括之前的博文)请评论留言,谢谢! 由于代码里注释的很清楚,我就不做过多的解释了. 下面是这次总结的头文件,用来放置一些类和方法. //C++运算符重载实例.h #pragma once #include <iostream> using namespace std; class chongzai { private: int i, j, k; public: chongzai() { i =

  • VisualStudio2019构建C/C++静态库和动态库dll的问题 附源码

    1. 静态库和动态库 1.1. 静态链接库 举个例子,假如你在编写一个C++工程,根据业务逻辑,这个工程需要用到一些工具类,例如集合操作的工具类(暂且叫他collection_utils),于是你直接定义一个collection_utils.h头文件和一个collection_utils.cpp文件,在头文件中写一些工具函数的定义,在cpp文件中写函数的实现逻辑:如下所示: //---------------collection_utils.h--------------------------

  • jQuery插件ImageDrawer.js实现动态绘制图片动画(附源码下载)

    ImageDrawer.js是一款可以实现动态绘制图片动画的jQuery插件.通过ImageDrawer.js插件,你可以制作在页面中绘制图片的动态过程,你可以控制绘制动画的持续时间等参数,非常有趣. 效果展示       源码下载 使用方法 使用该动态绘制图片插件需要在页面中引入imagedrawer.css,jquery和imagedrawer.js文件. <link rel="stylesheet" href="css/imagedrawer.css"

  • jQuery插件之Tocify动态节点目录菜单生成器附源码下载

    Tocify是一个能够动态生成文章节点目录的jQuery插件.假如我们有一篇很长的文章,文章有多个节点,那么使用Tocify可以根据节点元素动态生成文章目录,点击目录可以平滑滚动到对应的节点,当然当滚动页面时,目录结构会根据当前监听到的节点进行切换到当前目录状态. 效果展示         源码下载 Tocify目前支持Twitter Bootstrap和jQueryUI Themeroller两种主题风格,我们可以根据实际项目任选其中一种风格,另外必要条件jQuery 1.7.2+和jQuer

  • C++ Cmake的构建静态库和动态库详解

    目录 静态库和动态库的区别 构建示例 ADD_LIBRARY 同时构建静态和动态库 SET_TARGET_PROPERTIES 动态库的版本号 安装共享库和头文件 使用外部共享库和头文件 解决 :make后头文件找不到的问题 解决:找到引用的函数问题 特殊的环境变量CMAKE_INCLUDE_PATH和CMAKE_LIBRARY_PATH 总结 静态库和动态库的区别 1.静态库的扩展名一般为".a"或者".lib":动态库的扩展名一般为".so"

  • Android NDK生成及连接静态库与动态库的方法

    对于Android应用开发,大部分情况下我们使用Java就能完整地实现一个应用.但是在某些情况下,我们需要借助C/C++来写JNI本地代码.比如,在使用跨平台的第三方库的时候:为了提升密集计算性能的时候(这种情况下往往还可能会直接使用汇编语言).因此,这里我将为大家介绍如何给其它开发者创建可供使用的静态库或动态库.而应用开发者如何去连接这些生成的静态库或动态库.由于现在Android Studio已经比较成熟,因此以下描述将基于Android Studio的目录布局. 在Android Stud

  • Linux静态库与动态库实例详解

    Linux静态库与动态库实例详解 1. Linux 下静态链接库编译与使用 首先编写如下代码: // main.c #include "test.h" int main(){ test(); return 0; } // test.h #include<iostream> using namespace std; void test(); // test.c #include "test.h" void test(){ cout<< &quo

  • Linux下g++编译与使用静态库和动态库的方法

    在windows环境下,我们通常在IDE如VS的工程中开发C++项目,对于生成和使用静态库(*.lib)与动态库(*.dll)可能都已经比较熟悉,但是,在linux环境下,则是另一套模式,对应的静态库(*.a)与动态库(*.so)的生成与使用方式是不同的.刚开始可能会不适应,但是用多了应该会习惯这种使用,因为步骤上并没有VS下配置那么繁琐. 下面就分别总结下linux下生成并使用静态库与动态库的方法:(由于是C++项目,所以编译器用的g++,但是与gcc的使用是相通的) 首先是准备工作,把我们需

  • 深入探讨Linux静态库与动态库的详解(一看就懂)

    库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行.库分静态库和动态库两种. 一.静态库和动态库的区别1. 静态函数库这类库的名字一般是libxxx.a:利用静态函数库编译成的文件比较大--空间,因为整个函数库的所有数据都会被整合进目标代码中,他的优点就显而易见了,即编译后的执行程序不需要外部的函数库支持,因为所有使用的函数都已经被编译进去了.当然这也会成为他的缺点,因为如果静态函数库改变了,那么你的程序必须重新编译.2. 动态函数库这类库的名字一般是libxxx.so;相对于静态

  • xcode 详解创建静态库和动态库的方法

    xcode 创建静态库和动态库 1.linux中静态库和动态库区别: 库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行.库分静态库和动态库两种. 静态库:这类库的名字一般是libxxx.a:利用静态函数库编译成的文件比较大,因为整个函数库的所有数据都会被整合进目标代码中,他的优点就显而易见了,即编译后的执行程序不需要外部的函数库支持,因为所有使用的函数都已经被编译进去了.当然这也会成为他的缺点,因为如果静态函数库改变了,那么你的程序必须重新编译. 动态库:这类库的名字一般是lib

  • SpringBoot2.0+阿里巴巴Sentinel动态限流实战(附源码)

    Sentinel 是什么? 随着微服务的流行,服务和服务之间的稳定性变得越来越重要.Sentinel 以流量为切入点,从流量控制.熔断降级.系统负载保护等多个维度保护服务的稳定性. Sentinel 具有以下特征: 丰富的应用场景:Sentinel 承接了阿里巴巴近 10 年的双十一大促流量的核心场景,例如秒杀(即突发流量控制在系统容量可以承受的范围).消息削峰填谷.集群流量控制.实时熔断下游不可用应用等. 完备的实时监控:Sentinel 同时提供实时的监控功能.您可以在控制台中看到接入应用的

随机推荐