前言(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. 程序知道我可以链接谁(用谁的函数)
  2. 程序真的导入了相应的二进制文件

过程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的编译和链接的更多相关文章

  1. IDEA的maven项目中 静态文件编译的问题

    IDEA的maven项目中,默认源代码目录下的xml等资源文件并不会在编译的时候一块打包进classes文件夹,而是直接舍弃掉. 如果使用的是Eclipse,Eclipse的src目录下的xml等资源 ...

  2. Java之——Web项目中DLL文件动态加载方法

    本文转自:https://blog.csdn.net/l1028386804/article/details/53903557 在Java Web项目中,我们经常会用到通过JNI调用dll动态库文件来 ...

  3. IDEA的maven项目中静态文件编译的路径问题(未测试)

    转自:http://www.cnblogs.com/signheart/p/6625126.html IDEA的maven项目中,默认源代码目录下的xml等资源文件并不会在编译的时候一块打包进clas ...

  4. DelphiXE10.1项目中增加预编译的方法

    操作: 菜单选择Proceject->Options->Delphi Compilerz在Conditional Defines(第一行)中添加预编译标识.例:VCL代码:uses{$IF ...

  5. 在CLion项目中指定不同版本的链接库

    在项目中, 需要使用到libevent-2.1.x, 但是Ubuntu16.04自带的libevent版本为2.0.5, 需要另外编译安装新版的libevent, 安装过程很简单 -stable.ta ...

  6. Linux中程序的编译和链接过程

    1.从源码到可执行程序的步骤:预编译.编译.链接.strip 预编译:预编译器执行.譬如C中的宏定义就是由预编译器处理,注释等也是由预编译器处理的. 编译: 编译器来执行.把源码.c .S编程机器码. ...

  7. 怎么向Xcode6 IOS8之后向项目中添加预编译文件

    苹果的XCode在6版本之后新建项目时取消了自动创建预编译头文件pch,该文件里存放的工程中一些不常被修改的代码,比如常用的框架头文件,这样做的目的提高编译器编译速度.我们可以往里面加入一些项目中都要 ...

  8. vue项目中跳转到外部链接方法

    当我们在文件中,如果是vue页面中的内部跳转,可以用this.$router.push()实现,但是如果我们还用这种方法跳到外部链接,就会报错,我们一看链接的路径,原来是我们的外部链接前面加上了htt ...

  9. 解决cuvid中的sample编译和链接问题

    unzip Video_Codec_SDK_9.0.20.zip cd Video_Codec_SDK_9.0.20/Samples/AppDecode/AppDecImageProvider vi ...

  10. iOS项目中常见的文件

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

随机推荐

  1. redis 简单整理——redis shell[九]

    前言 简单介绍一下redis的shell命令. 正文 redis 提供了一些工具,如redis-cli.redis-server.redis-benchmark等. redis-cli -r 对red ...

  2. Python阿里云消息推送调用API

    很多公司测试APP推送时候,应该也是很头疼:推送环境:测试.正式,稍不注意就把测试的push到正式上,导致所有用户都收到 例子很多: 其实阿里.极光都有推送Api,直接调用API就ok,特别是有的公司 ...

  3. pyaudio音频录制python

    python3.7不支持pyaudio pip在线安装 whl下载地址:https://github.com/intxcc/pyaudio_portaudio/releases 下载后使用pip离线安 ...

  4. C#微服务必学清单

    在 C# 领域,有一些不错的微服务书籍和开源框架,对于学习微服务相关知识非常有帮助.以下是一些建议您阅读的微服务书目和开源框架. 微服务书目: 1. <Building Microservice ...

  5. 力扣289(java)-生命游戏(中等)

    题目: 根据 百度百科 , 生命游戏 ,简称为 生命 ,是英国数学家约翰·何顿·康威在 1970 年发明的细胞自动机. 给定一个包含 m × n 个格子的面板,每一个格子都可以看成是一个细胞.每个细胞 ...

  6. 3 种发布策略,解决 K8s 中快速交付应用的难题

    作者 | 郝树伟(流生)阿里云高级研发工程师 前言 软件技术更新换代很快,但我们追求的目标是一直不变的,那就是在安全稳定的前提下,增加应用的部署频率,缩短产品功能的迭代周期,这样的好处就是企业可以在更 ...

  7. MySQL 深潜 - 一文详解 MySQL Data Dictionary

    ​简介: 在 MySQL 8.0 之前,Server 层和存储引擎(比如 InnoDB)会各自保留一份元数据(schema name, table definition 等),不仅在信息存储上有着重复 ...

  8. [Blockchain] Cosmos Starport 101 - 为你的新数据类型 生成代码

    # 项目模板 $ starport app github.com/hello/planet --address-prefix your_new_prefix 项目目录结构的说明看这里: https:/ ...

  9. dotnet 8 WPF 支持在 RDP 远程桌面状态下启用渲染硬件加速

    本文将和大家介绍在 dotnet 8 里 WPF 引入的新功能之一,在 RDP 远程桌面状态下启用渲染硬件加速 在 dotnet 8 之前,在用户进行 RDP 远程桌面时 WPF 应用将默认关闭硬件渲 ...

  10. WPF 将 StaticResource 和 ResourceDictionary 放在一起的魔幻行为

    本文将记录一些在 WPF 里面,使用 StaticResource 将 ResourceDictionary 玩坏的做法.大家可以放心的是,这些玩法基本只有高级玩家或逗比开发者才会使用到 后加入的资源 ...