当前位置: 首页 > news >正文

C++高手进阶:Windows 模块加载的艺术与策略

前文我们讲到了怎么不依赖第三库,搭建自己的测试框架
没有看的读者可以通过这个链接自行阅读:
👉👉👉 自力更生:0依赖三方库,手把手教你打造专属C++测试框架
作为项目开发来说,我们通常会将不同的功能、数据进行分层或分块,这其实就涉及到不同模块之间的相互调用。选择什么样的策略来实现模块间能力的交互,其实是非常重要的。
本文中,我们就来抽象业务逻辑,看下模块间的加载是如何实现的。

文章目录

    • 模块间的加载方式
      • 动态加载(Dynamic Loading)
      • 静态加载(Static Loading)
    • 什么时候用什么方式
    • 怎么用
      • 动态加载示例介绍
      • 静态加载示例介绍
    • 总结

模块间的加载方式

通常而言,模块间的加载方式可以分为两类:

  • 动态加载
  • 静态加载

什么是动态加载呢?简单而言就是模块间遵循动态加载协议,平台层模块或业务层模块主动的调用 LoadLibrary()GetProcAddress接口来加载其他模块。

什么是静态加载呢?简单而言就是模块间遵循静态加载协议,平台层模块或业务层模块在编译时就将其他模块的依赖关系链接进来(通常建议在编写平台层 CMakeLists.txt 时,将依赖的库或模块列入链接库的列表,并设置链接类型为静态库)。

笔者在撰写此文时特地查了相关资料,对它们的特点做下介绍(PS: 文中有些地方可能有误,如有疑问,欢迎指正):

动态加载(Dynamic Loading)

动态加载指的是在程序运行时,根据需要加载和卸载库或模块。这种方式有以下几个特点:

  1. 运行时链接:库的代码在程序运行时才被加载,通常是通过动态链接库(Dynamic Link Library,DLL)在Windows上或共享库(Shared Object,SO)在Unix-like系统上实现。
  2. 灵活性:可以在不重启程序的情况下加载或卸载模块,提供更高的灵活性。
  3. 内存使用:只需要加载正在使用的库,节省内存。
  4. 更新和维护:可以独立更新库而不需要重新编译整个程序。
  5. 依赖管理:需要更复杂的依赖管理,因为库在运行时才被加载。

静态加载(Static Loading)

静态加载指的是在程序编译时,库或模块已经被链接到程序中。这种方式有以下几个特点:

  1. 编译时链接:库的代码在编译期间被链接到最终的可执行文件中。
  2. 加载时间:因为库的代码已经是程序的一部分,所以不需要在程序运行时加载。
  3. 内存使用:静态加载的库会占用更多的内存,因为所有库的代码都会被包含在最终的可执行文件中。
  4. 更新和维护:更新静态加载的库可能需要重新编译整个程序。
  5. 依赖管理:静态加载简化了依赖管理,因为所有依赖都包含在程序中。

什么时候用什么方式

也许读者在其他地方看到过它的使用建议,比如说根据加载的时间内存的使用模块间交互的频繁性模块间的依赖关系复杂度模块的更新频率等方面进行选择。

笔者就不在从这些维度进行阐述了。

笔者从工程实践的角度出发,讲讲两种使用的场景:

  1. 当主模块没有代码逻辑依赖该模块暴露的API接口时,该模块的能力仅仅是主模块能力的增强,就可以采用动态加载方式。
  2. 当主模块有代码逻辑依赖该模块暴露的API接口时,该模块的能力是主模块的核心功能,就可以采用静态加载方式。

怎么用

动态加载示例介绍

场景: 假设有主模块A,在其上开发了模块B,以进行能力增强。

那么对于模块 A 来说,得做如下事情:

  1. 定义模块间交互的协议类 TestModuleBase,用于模块 B 来派生,自定义自己的加载、卸载业务逻辑。
// xx.h
class TestModuleBase
{
public:TestModuleBase();virtual ~TestModuleBase();virtual SystemStatus initialize();virtual SystemStatus uninitialize();private:TestModuleBase(const TestModuleBase&) = delete;TestModuleBase& operator=(const TestModuleBase&) = delete;
};typedef TestModuleBase* (__cdecl* RxProlModule)();// xx.cpp
TestModuleBase::TestModuleBase()
{}TestModuleBase::~TestModuleBase()
{}SystemStatus TestModuleBase::initialize()
{return SystemStatus::e_Ok;
}SystemStatus TestModuleBase::uninitialize()
{return SystemStatus::e_Ok;
}
  1. 定义一个模块加载器 TestModuleLoader 类,用来管理模块的加载和卸载。
// xx.h
class DEMO_TEST_STATIC_EXPORT TestModuleLoader
{
public:TestModuleLoader() = delete;static SystemStatus loadModule(const wchar_t* pPath);static SystemStatus unloadModule(const wchar_t* pPath);private:TestModuleLoader(const TestModuleLoader&) = delete;TestModuleLoader& operator=(const TestModuleLoader&) = delete;
};// xx.cpp
SystemStatus TestModuleLoader::loadModule(const wchar_t* pPath)
{HMODULE pHandleModule = LoadLibrary(pPath);if (!pHandleModule == NULL) {return SystemStatus::e_Fail;}RxProlModule rxProlModule = (RxProlModule)(GetProcAddress(pHandleModule, "rxProtocalModule"));if (rxProlModule == NULL) {FreeLibrary(pHandleModule);return SystemStatus::e_Fail;}TestModuleBase* pModule = rxProlModule();if (pModule == NULL) {FreeLibrary(pHandleModule);return SystemStatus::e_Fail;}return pModule->initialize();
}SystemStatus TestModuleLoader::unloadModule(const wchar_t* pPath)
{if (pPath == NULL) {return SystemStatus::e_Fail;}HMODULE pHandleModule = GetModuleHandle(pPath);if (pHandleModule == NULL) {return SystemStatus::e_Fail;}RxProlModule rxProlModule = (RxProlModule)(GetProcAddress(pHandleModule, "rxProtocalModule"));if (rxProlModule == NULL) {FreeLibrary(pHandleModule);return SystemStatus::e_Fail;}TestModuleBase* pModule = rxProlModule();if (pModule == NULL) {FreeLibrary(pHandleModule);return SystemStatus::e_Fail;}SystemStatus uninitResult = pModule->uninitialize();if (uninitResult != SystemStatus::e_Ok) {return uninitResult;}FreeLibrary(pHandleModule);return SystemStatus::e_Ok;
}
  1. 在适当的时机,比如说模块A初始化和反初始化的时候,对模块B进行主动的加载和卸载。
SystemStatus PlatformModule::initialize()
{...auto ss = TestModuleLoader::loadModule("DemoModule1.dll")if (ss != SystenStatus::e_Ok) {return ss;}...
}SystemStatus PlatformModule::uninitialize()
{...auto ss = TestModuleLoader::unloadModule("DemoModule1.dll")if (ss != SystenStatus::e_Ok) {return ss;}...
}

那么对于模块 B 来说,得做如下事情:

  1. 派生协议类 TestModuleBase,实现自己的加载、卸载业务逻辑。
// xx.h
class DemoTestModule :public TestModuleBase
{
public:DemoTestModule();~DemoTestModule();SystemStatus initialize() override;SystemStatus uninitialize() override;private:DemoTestModule(const DemoTestModule&) = delete;DemoTestModule& operator=(const DemoTestModule&) = delete;bool m_initialized = false;
};DEMO_TEST_C_EXPORT TestModuleBase* rxProtocalModule();// xx.cpp
DEMO_TEST_C_EXPORT TestModuleBase* rxProtocalModule()
{static DemoTestModule module;return &module;
}DemoTestModule::DemoTestModule()
{}DemoTestModule::~DemoTestModule()
{}SystemStatus DemoTestModule::initialize()
{if (m_initialized) {return SystemStatus::e_NotSpecified;}m_initialized = true;// do something for initreturn SystemStatus::e_Ok;
}SystemStatus DemoTestModule::uninitialize()
{if (m_initialized) {// do uninit thingm_initialized = false;}return SystemStatus::e_Ok;
}

静态加载示例介绍

场景: 假设有主模块A,在其上开发了模块B,以实现核心功能。

  1. 对于模块B 来说,需定义一个模块初始化的类,实现该模块中对象的初始化和反初始化。同时给该类设置一个静态对象。
class DemoModuleInit
{
public:DemoModuleInit(){// do init thing}~DemoModuleInit(){// do uninit thing}
};static DemoModuleInit g_demoModuleInit;
  1. 在模块 A 的CMakeLists.txt中,将模块B的静态库或模块链接进来。
link_directories("${PROJECT_SOURCE_DIR}/lib")
...
link_libraries(DemoModule1)

总结

以上就是模块间加载的两种方式,动态加载和静态加载,以及它们的使用场景和示例。希望能对读者有所帮助。

如果读者有不理解的地方,欢迎私信交流。

相关文章:

C++高手进阶:Windows 模块加载的艺术与策略

前文我们讲到了怎么不依赖第三库,搭建自己的测试框架 没有看的读者可以通过这个链接自行阅读: 👉👉👉 自力更生:0依赖三方库,手把手教你打造专属C测试框架 作为项目开发来说,我们通常…...

基于STM32单片机老人体温心率血氧跌倒定位短信报警

一.硬件及设计功能 以STM32F103C8T6为中央处理器,GPS模块用采集数据,将数据发送给单片机后,单片机根据定位计算公式得出当前位置的经纬度信息和时间信息。经过LCD显示器处理后得出和时间信息SIM800模块发送短信到设定的手机号上,将…...

【测评】雨云香港三区云服务器,2核2G 5兆,仅需38元/月

写在前面 雨云香港三区云服务器,高性能的 AMD EPYC 处理器 企业级 NVME SSD 高性能云服务器。2核2G 10兆 400G防御,仅需38元/月,年付7折仅 319.2元/年。 官网:https://www.rainyun.com 本次测评服务器配置如下: C…...

如何应对Android面试官 -> 玩转 Fragment

前言 本章主要讲解下 Framgent 的核心原理; 基础用法 线上基础用法,其他的可以自行百度 FragmentManager manager getSupportFragmentManager(); FragmentTransaction transaction manager.beginTransaction(); transaction.add(R.id.contentlayout,…...

sdbusplus:通过文件描述符传递数据

有的时候需要传递大量的数据,如果将数据通过dbus传递,会消耗大量的带宽。可以通过传递一个文件描述符替代传递数据: 以下的service通过文件描述符接收数据: //fd_service.cpp #include <sdbusplus/asio/connection.hpp> #include <sdbusplus/asio/object_server…...

HyperLPR3 车牌识别

Linux 之前安装了python3 apt install python3.8-venv cd /root python3 -m venv HyperLPR3 REM cd HyperLPR3 source HyperLPR3/bin/activate 参考 https://www.jb51.net/article/222885.htm python -m pip install hyperlpr3 里面有fastapi&#xff0c;opencv等 错误&#x…...

面试的内容

1.C的三大特性&#xff1a;封装&#xff0c;继承&#xff0c;多态 2.C11的特性 3.NULL与Nullptr的区别: nullptr是一个特殊的空指针常量&#xff0c;不能被隐式转换为其他类型。 NULL 在一些情况下可能会发生隐式类型转换 4.智能指针 5.stl/Qt stl stl容器包含哪些&…...

剪映网页版

https://www.capcut.cn/web 免费&#xff0c;免安装&#xff0c;跨平台&#xff0c;视频云合成&#xff0c;简直太好用了&#xff01;...

pgsql

创建分区表&#xff1a; PostgreSQL分区表_pg分区表-CSDN博客 创建list分区的函数 create or replace function create_list_fq(tb_name char, row_name char) returns int AS $$ declares char; beginraise notice CREATE TABLE if not exists %_% PARTITION OF % FOR VALU…...

Kotlin学习笔记 泛型

在 Kotlin 中&#xff0c;T 通常用作类型参数的占位符&#xff0c;它在实例化或传递参数时会被替换成具体的类型。 Kotlin 支持泛型&#xff0c;这意味着您可以编写可以与多种数据类型一起工作的代码&#xff0c;而不必为每种数据类型编写单独的代码。 ### 泛型类和函数 在 …...

开发者必看:Linux终端的10大装逼神器,让你的命令行炫酷起来!

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…...

20 VUE学习:插件

介绍 插件 (Plugins) 是一种能为 Vue 添加全局功能的工具代码。下面是如何安装一个插件的示例&#xff1a; import { createApp } from vueconst app createApp({})app.use(myPlugin, {/* 可选的选项 */ })一个插件可以是一个拥有 install() 方法的对象&#xff0c;也可以直接…...

python之any用法

写法对比 代码一&#xff1a; any(i for i in [0,1]) 代码2&#xff1a; any([i for i in [0,1]]) 优劣 结论&#xff1a;代码一写法更好 解释&#xff1a; 在 Python 中&#xff0c;any() 函数可以接受一个可迭代对象作为参数&#xff0c;并返回 True 如果可迭代对象…...

【前端学习——react坑】useState使用

问题 使用useState 时&#xff0c;例如 const [selectedId, setSelectedId] useState([false,true,false]);这样直接利用&#xff0c;无法引发使用selectedId状态的组件的变化&#xff0c;但是selectedId是修改了的 let tempselectedId;temp[toggledId]selectedId[toggledId…...

【前端每日基础】day28——async/await

async/await 是ES2017&#xff08;ES8&#xff09;引入的用于处理异步操作的语法糖&#xff0c;基于Promise实现。它使得异步代码看起来像同步代码&#xff0c;从而提高了代码的可读性和可维护性。以下是对 async/await 的详细讲解。 基本语法 async 函数 在一个函数前加上 as…...

错误记录:从把项目从Tomcat8.5.37转到Tomcat10.1.7

错误信息&#xff1a;在本地Servlet项目里没有报错&#xff0c;但是浏览器跳转该servlet时报错 型 异常报告 消息 实例化Servlet类[com.wangdao.lx.MyServlet1]异常 描述 服务器遇到一个意外的情况&#xff0c;阻止它完成请求。 例外情况 jakarta.servlet.ServletExceptio…...

AJAX基础知识

定义 Ajax 异步 JavaScript 和 XML &#xff08; async javascript and xml &#xff09;&#xff0c;使用 Ajax 技术网页应用能够快速地将数据更新呈现在用户界面上&#xff0c;而不需要重载&#xff08;刷新&#xff09;整个页面&#xff0c;这使得程序能够更快地回应用户的操…...

xcode依赖包package已经安装,但是提示No such module ‘Alamofire‘解决办法

明明已经通过xcode自带的swift包管理器安装好了依赖包&#xff0c;但是却还是提示&#xff1a;No such module&#xff0c;这个坑爹的xcode&#xff0c;我也只能说服气&#xff0c;但是无奈&#xff0c;没办法攻打苹果总部&#xff0c;只能自己想解决办法了 No such module Ala…...

基于Centos7 安装k8s一主两从

一、资源准备 mac下虚拟机环境搭建 1、使用搜狐的iso源 http://mirrors.sohu.com/centos/7.5.1804/isos/x86_64/CentOS-7-x86_64-Minimal-1804.iso 下载 iso镜像。 2、https://www.macwk.com/soft/vmware 下载 mac vm虚拟机 3、搭建一主两从集群所需虚拟机 4、新建虚拟机…...

基于java实现图片中任意封闭区域识别

需求&#xff1a; 在浏览器中给用户呈现一张图片&#xff0c;用户点击图片中的某些标志物&#xff0c;需要系统给出标志物的信息反馈&#xff0c;达到一个交互的作用。 比如下图中&#xff0c;点击某个封闭区域时候&#xff0c;需要告知用户点击的区域名称及图形形状特性等等。…...

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇&#xff0c;在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下&#xff1a; 【Note】&#xff1a;如果你已经完成安装等操作&#xff0c;可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作&#xff0c;重…...

挑战杯推荐项目

“人工智能”创意赛 - 智能艺术创作助手&#xff1a;借助大模型技术&#xff0c;开发能根据用户输入的主题、风格等要求&#xff0c;生成绘画、音乐、文学作品等多种形式艺术创作灵感或初稿的应用&#xff0c;帮助艺术家和创意爱好者激发创意、提高创作效率。 ​ - 个性化梦境…...

MFC内存泄露

1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...

【论文笔记】若干矿井粉尘检测算法概述

总的来说&#xff0c;传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度&#xff0c;通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...

苍穹外卖--缓存菜品

1.问题说明 用户端小程序展示的菜品数据都是通过查询数据库获得&#xff0c;如果用户端访问量比较大&#xff0c;数据库访问压力随之增大 2.实现思路 通过Redis来缓存菜品数据&#xff0c;减少数据库查询操作。 缓存逻辑分析&#xff1a; ①每个分类下的菜品保持一份缓存数据…...

ETLCloud可能遇到的问题有哪些?常见坑位解析

数据集成平台ETLCloud&#xff0c;主要用于支持数据的抽取&#xff08;Extract&#xff09;、转换&#xff08;Transform&#xff09;和加载&#xff08;Load&#xff09;过程。提供了一个简洁直观的界面&#xff0c;以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...

uniapp微信小程序视频实时流+pc端预览方案

方案类型技术实现是否免费优点缺点适用场景延迟范围开发复杂度​WebSocket图片帧​定时拍照Base64传输✅ 完全免费无需服务器 纯前端实现高延迟高流量 帧率极低个人demo测试 超低频监控500ms-2s⭐⭐​RTMP推流​TRTC/即构SDK推流❌ 付费方案 &#xff08;部分有免费额度&#x…...

Element Plus 表单(el-form)中关于正整数输入的校验规则

目录 1 单个正整数输入1.1 模板1.2 校验规则 2 两个正整数输入&#xff08;联动&#xff09;2.1 模板2.2 校验规则2.3 CSS 1 单个正整数输入 1.1 模板 <el-formref"formRef":model"formData":rules"formRules"label-width"150px"…...

AspectJ 在 Android 中的完整使用指南

一、环境配置&#xff08;Gradle 7.0 适配&#xff09; 1. 项目级 build.gradle // 注意&#xff1a;沪江插件已停更&#xff0c;推荐官方兼容方案 buildscript {dependencies {classpath org.aspectj:aspectjtools:1.9.9.1 // AspectJ 工具} } 2. 模块级 build.gradle plu…...

rnn判断string中第一次出现a的下标

# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...