C++工作笔记-动态库中的单例类存储方式

背景

最近在改一个老C++大型项目,里面的动态库调用错综复杂,因为要加功能,发现各个动态库调用依靠各自导出的单例类进行,因为需要交互一些数据,特改造了此单例类,但发现没有实现业务效果,通过打印调试等,发现此单例类,构造函数运行了多次。此时就发现问题了。为什么会调用多个函数,并且打印的this指针地址也不同。特意记录了本文章。

感觉工作8年了,才接触到不同dll中创建同一单例类,地址是否一致,这种问题。感觉自己有点low啊。本科期间操作系统或计算机组成稍微学好一点也不至于花了半天时间,才发现这个问题。

结论

每个 DLL/SO 作为独立的加载模块,拥有专属且隔离的全局 / 静态存储区,这是单例实例地址不同的核心原因。

一个运行中的程序(进程)会有一个统一的虚拟地址空间(比如 32 位程序是 4GB),但这个地址空间会被 "分割" 给不同的加载单元(主程序、DLL/SO)。关键在于:

每个 DLL/SO 加载时,会在进程虚拟地址空间中占据独立的一段区域,并且拥有自己的:

代码段(.text):存放函数实现;

数据段(.data):存放已初始化的全局 / 静态变量;

BSS 段(.bss):存放未初始化的全局 / 静态变量。

这些段是模块私有的------ 也就是说,模块 A 的静态变量和模块 B 的同名静态变量,会被分配到进程虚拟地址空间的不同位置,彼此完全隔离。

如在 Windows 下用 VS 调试时:

  1. 主程序的单例实例地址可能是 0x00405000(主程序的静态存储区范围);
  2. DLL1 的单例实例地址可能是 0x10005000(DLL1 的静态存储区范围);
  3. DLL2 的单例实例地址可能是 0x20005000(DLL2 的静态存储区范围);

这些地址分属不同模块的内存区间,本质就是因为每个模块的静态存储区是隔离的。

所以如果是同一个程序常驻动态库中的线程想共享一些数据,还是用最常见的方式把:数据落地、管道、共享内存等都是可以的,看自己业务需求吧。

相关推荐
卷无止境5 小时前
Eigen 库如何借助 OpenMP 加速计算
c++·后端
卷无止境6 小时前
OpenMPI、MPICH 与 OpenMP:关系、核心概念与架构全解
c++·后端
郝学胜_神的一滴1 天前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake
卷无止境3 天前
C++ 的Eigen 库全解析
c++
卷无止境3 天前
现代 C++特性大盘点:一门脱胎换骨的老语言
c++·后端
郝学胜_神的一滴3 天前
CMake 27:缓存变量的特性、语法、类型与实操全解
c++·cmake
博客18005 天前
酷宝的使用方法,超好用的免费界面库,C++、MFC可用
c++·mfc·界面库·库来帮·酷宝
郝学胜_神的一滴5 天前
CMake 026:属性体系精讲、四大作用域全解 & 实战代码落地
c++·cmake
众少成多积小致巨6 天前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
LinXunFeng7 天前
Obsidian - 使用 Share Note 分享笔记并自部署
前端·笔记·github