CMake 进行多项目中dll的编译和链接
前言(maybe废话)
最近正在学习cherno的游戏引擎教程,他使用的是vs进行构建的,后面换了premake。而我用的是vscode+cmake,所以在构建整个项目的时候踩了不少的坑,也找了很多资料去努力解决,比如b站双笙子大佬的cmake教程(强推)。
遂有感而发,写下本篇博客记录一下。
duan20030920@gmail.com 这是我的邮箱,如果有朋友有什么疑问的话,欢迎咨询。当然也可以评论。
CMake是自由的
众所周知的是,CMake是基于你文件系统中的CMakeLists.txt文件的,所以,只要你文件夹里面有个CMakeLists文件,你都可以把该文件夹构建成一个项目。
由此而发,我们很自然地会想,我是否能在一个项目的文件夹中去再构建一个项目呢?
当然是可以的,只要你放CMakeLists,我们就是兄弟(雾)。
扯远了,咳咳。这篇博客主要是用来讲dll的。
说起dll,大家肯定都不陌生,它是windows平台下,用于动态链接的库文件。
什么是动态链接我在此就不多做赘述。
那么,当我们把一个项目编译成dll之后,我们该如何使用它呢?
是不是只要和.lib一样,target_link一下就行了呢?当然不是,win下的动态链接机制较为复杂,大体可以分成两个部分:
- 程序知道我可以链接谁(用谁的函数)
- 程序真的导入了相应的二进制文件
过程1
你可能会想,为什么我直接把程序链接上dll之后,不能直接使用呢?
如果你是高贵的linux用户,这么做是没问题的(.so)。
但是在window下,你不能直接这么做。
前面我们提到,exe是按需将dll中的函数和数据(以下简称“数据”)链接到其中的,所以我们肯定不能一口气将dll全部复制到程序中,那不就成了lib了。所以我们就需要知道我们要链接哪些数据,以及 我们到哪里链接哪些数据。
前者在我们通过引入dll的头文件解决,但问题是如果没有后者,即使我们知道要用谁,也是束手无策的。
所以windows就在它的dll动态链接系统中,提供了下面一种架构,在创建dll的同时,创建一个lib,这个lib负责告知exe,dll里面都有什么数据,相当于一个声明,所以它的size很小,也满足我们灵活性的需求。
(更加细节的部分,请看下方的“为何Windows下链接dll还需要一个.lib?)
过程2
与lib直接将数据复制到exe不同,dll是在运行时,将所需要的数据加载到内存的方式,提供给exe使用,并且实现了多个exe的复用。所以很多大型的项目都趋向于将库做成dll的形式与exe分离,减小exe的大小。
所以,我们的exe能不能找到dll就很关键。
于是我们对构建系统进一步发展,设计出dll的两种链接到exe的方式。它们分别是:隐式链接和显式链接。
所谓隐式与显式之别,就是 是否在程序中指定链接dll。
隐式链接
就是我们之前提到的,利用.lib和.h导入我们需要的信息,然后进行dll中的动态链接。
显式链接
在程序中指定,这需要window提供的库函数来实现。
# Dynamic-Link库函数
这是更底层的,线程级对dll加载的控制。
同时,在某种程度上,实现了延时加载和按需加载,也给了我们dll路径更多的选择。
是目前超大型项目的不二之选。
这个我们以后再细聊。
more information...
为何.exe运行时需要.dll在身边
首先要明确的是,dll是在我们的exe运行时动态的链接到程序中的,也就是说,等你要了,我才把我的二进制文件给你。
又因为cmake是作用于项目的构建时,所以它不能设定我们的应用程序去搜索指定目录下的dll。
windows下dll动态链接系统设定要求,在进行动态链接时,运行时程序会搜索如下路径的dll文件:
- exe路径下
- 系统变量path路径下
- system路径,也就是系统目录下
所以在 默认情况下,我们自己项目的dll文件需要放在exe目录下,那有没有其它的解决方案呢?
使用Window.h库函数动态加载dll
参考:
# c++ 分目录加载dll依赖
# exe与引用的dll不在同一目录下
既然它是运行时加载,那我把它写到我的程序里不就行了吗?
但是这种方案也会带来很多不便,比如说它需要我们修改源代码,而且不能使用模糊搜索,优雅,但没有完全优雅。
不过用于组织我们的项目可以说是绰绰有余了。
为何Windows下链接dll还需要一个.lib?
在 Windows 系统中使用 DLL(动态链接库)时需要一个额外的 .lib 文件作为导入库,主要是由于 Windows 链接模型和 DLL 如何被操作系统和应用程序使用的特定设计决策。以下是一些关键原因和这种设计的详细解释:
1. 链接过程的分隔
在 Windows 上,应用程序通常在编译时静态链接到库,或者在运行时动态链接。DLL 为动态链接提供支持,它们在应用程序启动后或运行中按需加载。然而,为了在编译时让链接器知道哪些函数可用,以及它们在 DLL 中的具体位置,就需要一个额外的导入库(.lib 文件)来作为中介:
- 导入库包含存根:导入库
.lib包含了指向 DLL 中函数的跳转存根(stubs)。这些存根提供了必要的信息,以确保在应用程序调用 DLL 中的函数时,能够正确地重定向到相应的函数实现。
2. 解析外部符号
当你的程序使用 DLL 中的函数时,编译器需要解析这些外部函数调用的位置。导入库包含了所有公开函数的符号及其入口点,这些信息在编译时被链接器使用,以确保运行时能够找到并调用正确的代码。
- 避免编译时直接依赖 DLL:如果没有导入库,编译器在编译时会需要直接访问 DLL,这在多数开发和构建环境中是不可行的。使用导入库简化了构建过程,允许编译器和链接器在没有实际 DLL 文件的情况下完成其工作。
3. 模块化和部署灵活性
使用导入库和 DLL 分离的方式提高了应用程序的模块化和部署灵活性。开发者可以仅重新部署修改过的 DLL 而不需要重新编译整个应用程序,这样可以方便地更新和维护大型应用。
- 允许不同的语言和工具链开发:DLL 可以由不同的编程语言编写和编译,只要它们遵循相同的调用约定。这样,C#、Visual Basic、C++ 等不同语言编写的组件都可以使用相同的 DLL,而通过导入库与之交互。
4. 运行时加载
DLL 可以在应用程序运行时加载和卸载,提供了高度的灵活性。导入库使得这种动态加载成为可能,因为它把编译时的外部依赖转换为运行时的动态查找和链接。
总的来说,导入库 .lib 在 Windows 系统中的使用提供了一个高效、灵活的机制,使得程序在编译时不必绑定特定的库实现,同时在运行时可以利用 DLL 提供的动态链接和模块化的好处。
CMake 进行多项目中dll的编译和链接的更多相关文章
- IDEA的maven项目中 静态文件编译的问题
IDEA的maven项目中,默认源代码目录下的xml等资源文件并不会在编译的时候一块打包进classes文件夹,而是直接舍弃掉. 如果使用的是Eclipse,Eclipse的src目录下的xml等资源 ...
- Java之——Web项目中DLL文件动态加载方法
本文转自:https://blog.csdn.net/l1028386804/article/details/53903557 在Java Web项目中,我们经常会用到通过JNI调用dll动态库文件来 ...
- IDEA的maven项目中静态文件编译的路径问题(未测试)
转自:http://www.cnblogs.com/signheart/p/6625126.html IDEA的maven项目中,默认源代码目录下的xml等资源文件并不会在编译的时候一块打包进clas ...
- DelphiXE10.1项目中增加预编译的方法
操作: 菜单选择Proceject->Options->Delphi Compilerz在Conditional Defines(第一行)中添加预编译标识.例:VCL代码:uses{$IF ...
- 在CLion项目中指定不同版本的链接库
在项目中, 需要使用到libevent-2.1.x, 但是Ubuntu16.04自带的libevent版本为2.0.5, 需要另外编译安装新版的libevent, 安装过程很简单 -stable.ta ...
- Linux中程序的编译和链接过程
1.从源码到可执行程序的步骤:预编译.编译.链接.strip 预编译:预编译器执行.譬如C中的宏定义就是由预编译器处理,注释等也是由预编译器处理的. 编译: 编译器来执行.把源码.c .S编程机器码. ...
- 怎么向Xcode6 IOS8之后向项目中添加预编译文件
苹果的XCode在6版本之后新建项目时取消了自动创建预编译头文件pch,该文件里存放的工程中一些不常被修改的代码,比如常用的框架头文件,这样做的目的提高编译器编译速度.我们可以往里面加入一些项目中都要 ...
- vue项目中跳转到外部链接方法
当我们在文件中,如果是vue页面中的内部跳转,可以用this.$router.push()实现,但是如果我们还用这种方法跳到外部链接,就会报错,我们一看链接的路径,原来是我们的外部链接前面加上了htt ...
- 解决cuvid中的sample编译和链接问题
unzip Video_Codec_SDK_9.0.20.zip cd Video_Codec_SDK_9.0.20/Samples/AppDecode/AppDecImageProvider vi ...
- iOS项目中常见的文件
html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...
随机推荐
- caidao qsnctfwp
进入网页发现如下内容 直接使用蚁剑连接 连接并进入后,在根目录下发现名为 flag 的文件,即可获取 flag -End-
- 【5】Spring IoC介绍
有部分 Java 开发者对 IoC(Inversion Of Control)和 DI(Dependency Injection)的概念有些混淆,认为二者是对等的. IoC 其实有两种方式,一种就是 ...
- mysql 必知必会整理—sql 简单语句[二]
前言 简单整理一下sql 排序与过滤. 正文 在这里需要创建一下一个数据库实例. 为了方便直接用docker 创建一下啊,方便简洁. https://hub.docker.com/_/mysql 按照 ...
- Django3.0连接数据库注意点
需先在应用下的__Init__.py文件中配置 import pymysqlpymysql.version_info=(1, 3, 13, 'final', 0) # 3.0时需要pymysql.in ...
- Fatal: (vsim-3381) obsolete library format 解决办法
有很多软件仿真都是black boxs 黑盒子模式,所以用modelsim提示该错误.错误的原因是在于库用了其他旧的软件版本编译好的. 所以解决的办法如下: 在已经映射好的库选择refresh就可以刷 ...
- ORA-01555:snapshot too old: rollback segment number X with name "XXXX" too small
ORA-01555:snapshot too old: rollback segment number X with name "XXXX" too small 在查询快照的时候 ...
- dotnet 5 从 IL 层面分析协变返回类型新特性
在 C# 9.0 里面添加的一个新特性是支持协变返回类型,也就说子类重写了基类的抽象或虚拟方法,可以在返回值里面返回协变的类型,也就是返回值的类型可以是继承原本子类返回值类型的子类.本文将来从 IL ...
- dotnet 在 UOS 国产系统上安装 dotnet sdk 的方法
本文告诉大家如何在 UOS 国产系统上安装 dotnet sdk 的方法 使用的 UOS 是 UOS 20 x64 版本,这个系统版本是基于 debian 10 的,可以使用 debian 10 的方 ...
- Linux系统命令-目录命令
1.ls命令:主要作用是显示目录下的内容 基本格式 [root@localhost ~]# ls [选项] [参数是文件名或目录名] 常用选项 -a:显示所有文件 --color=when:支持颜色输 ...
- 超轻量级的c#版基于文件的日志记录工具,可定制输出格式,可指定日志文件
这是我自己个人编写的日志记录,主要使用在只需要记录日志,偶尔到文件中查看一下日志记录的情况.我自己写的一些服务之类的是使用了这个的,代码很少,使用很简单. 第一步 搜索和安装我的Nuget包 搜索和安 ...