DoubleLi

qq: 517712484 wx: ldbgliet

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::
  4737 随笔 :: 2 文章 :: 542 评论 :: 1615万 阅读
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

现象描述

在一个动态库里使用了std::map和std::set,在windows上用VS2010调试一切正常。但在linux下,用的是Ubuntu64位虚拟机调试,编译链接都OK,但只要调到这个so库里的使用std::map和std::set的地方,这些变量在qt调试里都是**<无法访问>,而且只要调用这些变量的非size()接口,比如find()、insert()等接口都会导致程序崩溃**。
郁闷之间。。。。。。

定位之路

1、和windows调用比较,没发现任何问题

2、修改cmakelists.txt,发现也没有什么可以改的,能改的怎么改结果都一样

3、最笨的办法之一用上,写一段这样的代码:

typedef map<string, void*> STRMAP;
STRMAP sm;
string str = “sdfasf”;
STRMAP::iterator it = sm.find(str);
sm.insert(std::make_pair(“fasf”, &sm));
it = sm.find(str);
放到可执行程序里调用,没有任何问题;放到so里的接口里的开始,按说单步到string str = “sdfasf”;时,sm应该被初始化,变量里能看到sm的值,但实际上sm的值就是<无法访问>,再往下执行find()这一句,就崩溃了。

4、面向互联网大法编程,百度、微软必应(不FQ,不能狗狗),发现一个帖子,缩小有问题代码的范围,于是

5、最笨的办法之二用上,缩小so代码范围

so之保留一个接口,接口里之保留上面的那一段代码,sm值可见了,find()、insert()接口可以正常调用;
再一点一点把代码还原,终于定位到了问题

std::map、std::set崩溃的终极原因

经过上述各种定位,耗时106060秒,终于定位到令人吐血的原因:
so里#include了一个某标准头文件,该头文件里在结构体定义的前面有一句#pragma pack(x),但后面并没有还原,于是前后分别加上#pragma pack(push)、#pragma pack(pop),单步时变量变得可见,调用std::map和std::set的find()、insert()等方法恢复正常,不再崩溃!
为了确认问题,把加的#pragma pack(push)、#pragma pack(pop)注掉,问题立马复现,加上立马变好。
为什么会这样?so里有很多代码文件,有的没有引用这个头文件,但直接#include < map >
#include < set >,有一个cpp文件是先引用了这个标准头文件,而后再#include < map >
#include < set >,导致同一个so里map、set字节对齐不一致。但为什么windows下正确,但linux下却异常崩溃?

posted on   DoubleLi  阅读(448)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
历史上的今天:
2018-07-10 微服务架构的核心要点和实现原理
2017-07-10 VS2015编译Boost1.64
2017-07-10 boost库生成文件命名和编译
2015-07-10 linux中的cd ..和cd -命令有什么区别?
2012-07-10 CString的GetBuffer用法,GetBuffer本质,GetBuffer常见问题解决方法 .
2012-07-10 GetSystemMetrics()函数的用法
2012-07-10 _itoa atoi、atof、itoa、itow _itoa_s 类型转换使用说明
点击右上角即可分享
微信分享提示