线程中如何使用对象_多线程中如何使用gdb精确定位死锁问题

bc14cda2aa408d9de7e745b6dbac7b19.gif

在多线程开发过程中很多人应该都会遇到死锁问题,死锁问题也是面试过程中经常被问到的问题,这里介绍在c++中如何使用gdb+python脚本调试死锁问题,以及如何在程序运行过程中检测死锁。

首先介绍什么是死锁,看下维基百科中的定义:

死锁(英语:Deadlock),又译为死结,计算机科学名词。当两个以上的运算单元,双方都在等待对方停止运行,以获取系统资源,但是没有一方提前退出时,就称为死锁。在多任务操作系统中,操作系统为了协调不同行程,能否获取系统资源时,为了让系统运作,必须要解决这个问题。

维基百科中介绍的是进程死锁,多线程中也会产生死锁,一样的道理,这里不作过多介绍。

死锁的四个条件

  • 禁止抢占(no preemption):系统资源不能被强制从一个进程(线程)中退出,已经获得的资源在未使用完之前不能被抢占。

  • 等待和保持(hold and wait):一个进程(线程)因请求资源阻塞时,对已获得的资源保持不放。

  • 互斥(mutual exclusion):资源只能同时分配给一个进程(线程),无法多个进程(线程)共享。

  • 循环等待(circular waiting):一系列进程(线程)互相持有其他进程(线程)所需要的资源。

只有同时满足以上四个条件,才会产生死锁,想要消除死锁只需要破坏其中任意一个条件即可。

如何调试多线程死锁问题

多线程出现死锁的大部分原因都是因为多个线程中加锁的顺序不一致导致的,看如下这段会出现死锁的代码:

using std::cout;std::mutex mutex1;std::mutex mutex2;std::mutex mutex3;void FuncA() {   std::lock_guard<std::mutex> guard1(mutex1);   std::this_thread::sleep_for(std::chrono::seconds(1));   std::lock_guard<std::mutex> guard2(mutex2);   std::this_thread::sleep_for(std::chrono::seconds(1));}void FuncB() {   std::lock_guard<std::mutex> guard2(mutex2);   std::this_thread::sleep_for(std::chrono::seconds(1));   std::lock_guard<std::mutex> guard3(mutex3);   std::this_thread::sleep_for(std::chrono::seconds(1));}void FuncC() {   std::lock_guard<std::mutex> guard3(mutex3);   std::this_thread::sleep_for(std::chrono::seconds(1));   std::lock_guard<std::mutex> guard1(mutex1);   std::this_thread::sleep_for(std::chrono::seconds(1));}int main() {   std::thread A(FuncA);   std::thread B(FuncB);   std::thread C(FuncC);   std::this_thread::sleep_for(std::chrono::seconds(5));   if (A.joinable()) {       A.join();  }   if (B.joinable()) {       B.join();  }   if (C.joinable()) {       C.join();  }   cout << "hello\n";   return 0;}

如图:

a9104440c2c3eec54c69bc4fbc1b7f57.png

  • 线程A已经持有mutex1,想要申请mutex2,拿到mutex2后才可以释放mutex1和mutex2,而此时mutex2被线程B占用。

  • 线程B已经持有mutex2,想要申请mutex3,拿到mutex3后才可以释放mutex2和mutex3,而此时mutex3被线程C占用。

  • 线程C已经持有mutex3,想要申请mutex1,拿到mutex1后才可以释放mutex3和mutex1,而此时mutex1被线程A占用。

三个线程谁也不让着谁,导致了死锁。

传统gdb调试多线程死锁方法

(1)attach id关联到发生死锁的进程id

(gdb) attach 109Attaching to process 109[New LWP 110][New LWP 111][New LWP 112][Thread debugging using libthread_db enabled]Using host libthread_db library "/lib/x86_64-linux-gnu/libthread_db.so.1".0x00007fa33f9e8d2d in __GI___pthread_timedjoin_ex (threadid=140339109693184, thread_return=0x0, abstime=0x0,block=out>) at pthread_join_common.c:89      pthread_join_common.c: No such file or directory.

(2)info threads查看当前进程中所有线程的信息,也可以查看到部分堆栈信息

(gdb) info threadsId   Target Id         Frame* 1    Thread 0x7fa33ff10740 (LWP 109) "out" 0x00007fa33f9e8d2d in __GI___pthread_timedjoin_ex (threadid=140339109693184, thread_return=0x0, abstime=0x0, block=) at pthread_join_common.c:892    Thread 0x7fa33ec80700 (LWP 110) "out" __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:1353    Thread 0x7fa33e470700 (LWP 111) "out" __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:1354    Thread 0x7fa33dc60700 (LWP 112) "out" __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135

这里可以看到2、3、4线程都在lock_wait状态,基本上可以看出或许是否问题,但是不一定,这里需要多次info threads看看这些线程有没有什么变化,多次如果都没有变化那基本上就是发生了死锁。

(3)thread id进入具体线程

(gdb) thread 2[Switching to thread 2 (Thread 0x7fa33ec80700 (LWP 110))]#0 __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135135     ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S: No such file or directory.

(4)bt查看当前线程堆栈信息

(gdb) bt#0 __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135#1 0x00007fa33f9ea023 in __GI___pthread_mutex_lock (mutex=0x7fa340204180 ) at ../nptl/pthread_mutex_lock.c:78#2 0x00007fa340000fff in __gthread_mutex_lock(pthread_mutex_t*) ()#3 0x00007fa3400015b2 in std::mutex::lock() ()#4 0x00007fa3400016d8 in std::lock_guard<:mutex>::lock_guard(std::mutex&) ()#5 0x00007fa34000109b in FuncA() ()#6 0x00007fa340001c07 in void std::__invoke_impl(std::__invoke_other, void (*&&)()) ()

调试到这里基本已经差不多了,针对pthread_mutex_t却可以打印出被哪个线程持有,之后再重复步骤3和4,就可以确定哪几个线程以及哪几把锁发生的死锁,而针对于std::mutex,gdb没法打印具体的mutex的信息,不能看出来mutex是被哪个线程持有,只能依次进入线程查看堆栈信息。

然而针对于c++11的std::mutex有没有什么好办法定位死锁呢?

有。

可以算作第五步,继续:

(5)source加载deadlock.py脚本

(gdb) source -v deadlock.pyType "deadlock" to detect deadlocks.

(6)输入deadlock检测死锁

(gdb) deadlock[Switching to thread 3 (Thread 0x7f5585670700 (LWP 123))]#0 __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135135     in ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S[Switching to thread 4 (Thread 0x7f5584e60700 (LWP 124))]#0 __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135135     in ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S[Switching to thread 2 (Thread 0x7f5585e80700 (LWP 122))]#0 __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135135     in ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S#1 0x00007f5586bea023 in __GI___pthread_mutex_lock (mutex=0x7f5587404180 ) at ../nptl/pthread_mutex_lock.c:78[Switching to thread 3 (Thread 0x7f5585670700 (LWP 123))]#0 __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135#1 0x00007f5586bea023 in __GI___pthread_mutex_lock (mutex=0x7f55874041c0 ) at ../nptl/pthread_mutex_lock.c:78[Switching to thread 4 (Thread 0x7f5584e60700 (LWP 124))]#0 __lll_lock_wait () at ../sysdeps/unix/sysv/linux/x86_64/lowlevellock.S:135#1 0x00007f5586bea023 in __GI___pthread_mutex_lock (mutex=0x7f5587404140 ) at ../nptl/pthread_mutex_lock.c:78Found deadlock!Thread 2 (LWP 122) is waiting on pthread_mutex_t (0x00007f5587404180) held by Thread 3 (LWP 123)Thread 3 (LWP 123) is waiting on pthread_mutex_t (0x00007f55874041c0) held by Thread 4 (LWP 124)Thread 4 (LWP 124) is waiting on pthread_mutex_t (0x00007f5587404140) held by Thread 2 (LWP 122)

直接看结果,脚本检测出了死锁,并指明了具体的哪几个线程造成的死锁,根据输出信息可以明显看出来线程锁形成的环造成了死锁,找到了具体是哪几个线程构成的死锁环,就可以查看相应线程的堆栈信息查看到哪几把锁正在等待。

死锁检测脚本的原理:

a9104440c2c3eec54c69bc4fbc1b7f57.png还是拿上面图举例:

  • 线程A已经持有mutex1,想要申请mutex2,拿到mutex2后才可以释放mutex1和mutex2,而此时mutex2被线程B占用。

  • 线程B已经持有mutex2,想要申请mutex3,拿到mutex3后才可以释放mutex2和mutex3,而此时mutex3被线程C占用。

  • 线程C已经持有mutex3,想要申请mutex1,拿到mutex1后才可以释放mutex3和mutex1,而此时mutex1被线程A占用。

3c5df84bafc4f31daeae0cd82b237a52.png如图,三个线程形成了一个环,死锁检测就是检查线程之间是否有环的存在。单独检查死锁的环比较容易,这里延申下还涉及到简单环的概念,因为正常检测出来的环可能是个大环,不是权值顶点数最少的环,如果检测的环的顶点数较多,加大定位的代价,脚本就是检测的简单环,这里涉及到强连通分量算法简单环算法,比较繁琐就不过多介绍了,脚本来源于facebook的folly库(这里推荐看下google的abseil和facebook的folly,都是好东西),代码较长在文中不好列出,如果有需要的话可以自行下载或者关注加我好友发给你。

如何在代码中检测死锁

和上面介绍的原理相同,在线程加锁过程中始终维护一张图,记录线程之间的关系

A->B, B->C, C->A

然后在图中检出简单环,找到哪几个线程处在死锁状态,做好状态记录,就可以完备确认是具体是哪几个线程哪几把锁发生的死锁,代码同样比较长,可以关注加我好友哦~

b19f5d908c717c58c99046ddcebe24e4.png

REVIEW

◆◆往期回顾一文让你搞懂设计模式RAII妙用之计算函数耗时深入浅出虚拟内存C++线程池的实现之格式修订版

关于GDB你需要知道的技巧

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/261859.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

html中如何把两行合并单元格,css合并两列单元格内容

用纯DIVCSS做一个两行两列的表格&#xff0c;但第二列中两行怎么在html里把一行中的两列合并世界最不可以相信的话,就是从女人嘴里说出的话〃如上图&#xff0c;怎么做出上面图中的效果&#xff0c;分享大神详解CSS表格单元格占两行可以参考以下的代码&#xff1a; 单元格占两行…

ASP.NET MVC 实现二级域名(泛域名)

自从微软发布 ASP.NET MVC 和routing engine (System.Web.Routing)以来&#xff0c;就设法让我们明白你完全能控制URL和routing&#xff0c;只要与你的application path相结合进行扩展&#xff0c;任何问题都迎刃而解。如果你需要在所处的域或者子域处理数据标记的话&#xff0…

list和tuple

2019独角兽企业重金招聘Python工程师标准>>> list Python内置的一种数据类型是列表&#xff1a;list。list是一种有序的集合&#xff0c;可以随时添加和删除其中的元素。 比如&#xff0c;列出班里所有同学的名字&#xff0c;就可以用一个list表示&#xff1a; >…

springboot数据源不正确_Spring MVC 到 Spring Boot 的简化之路

Spring全家桶笔记&#xff1a;SpringSpring BootSpring CloudSpring MVC​shimo.im01 背景从Servlet技术到Spring和Spring MVC&#xff0c;开发Web应用变得越来越简捷。但是Spring和Spring MVC的众多配置有时却让人望而却步&#xff0c;相信有过Spring MVC开发经验的朋友能深刻…

MapXtreme 包含所有自带坐标系一览

CoordSys 对象包含关于 X 和 Y 坐标如何与其在 Earth 上的位置相关联的基本信息。 每个 Geometry 或 Map 对象都有一个关联的坐标系。 CoordSys 对象包含对坐标系的详细说明。 CoordSysFactory 类提供了各种用于创建不同 CoordSys 对象的方法。 所有 CoordSys 对象都是只读的&a…

如何学习streamdecoder类_2019年终巨献:一份拿下了阿里、网易、滴滴等大厂offer的学习笔记...

2019仅剩最后二十天&#xff0c;回顾今年初遇“寒冬”时&#xff0c;自己也挺慌的&#xff0c;但是经历过这么多次面试后&#xff0c;我才“醒悟”&#xff0c;所谓的“寒冬”&#xff0c;“冻死”的都是“衣服穿的少的”。年末了在这里做一个年度总结&#xff0c;今年面试了不…

自定义控件的构建(12)

Share 前面讲了模板的构建&#xff0c;我们忽略了一个细节&#xff0c;如果接触ASP.NET时间不长的话&#xff0c;一般都会看到数据表达式是<%#Eval(‘Name’)%>这种形式的&#xff0c; 那么我们为什么用<%#Container.Name%>这种形式呢&#xff0c;其实前者是ASP.NE…

insertAfter()

<div id"b">bbbbbbbbb</div> <div>dddddd</div> JavaScript window.οnlοadfunction(){var a document.createElement("span");var b document.createTextNode("cssrain");a.appendChild(b);var mubiao document.getE…

学计算机等级考试电脑版软件,计算机二级考试宝典电脑版

计算机二级考试宝典电脑版是一款专业的二级计算机内容学习软件。该软件由武汉大学团队真情研发&#xff0c;软件包含选择题1600道&#xff0c;非选择题109套&#xff0c;成功实现了考点和重点的全面覆盖式学习目的&#xff0c;对学生们学习起到了巨大的帮助。该版本是通过安卓模…

mysq进阶

学习资料&#xff1a; 官方文档&#xff1a;http://dev.mysql.com/doc/refman/5.0/en/tutorial.html 1.存储过程&#xff1a; 优点&#xff1a;业务逻辑封装在存储过程中&#xff0c;容易维护&#xff0c;执行效率也高。 缺点&#xff1a;不同的数据库功能函数等不一样&#xf…

HTML使用vue的 event,vue-js 特殊变量$event常识

背景如果我们要阻止默认事件&#xff0c;在 chrome 等浏览器中&#xff0c;我们可能要写一个&#xff1a;event.preventDefault();而在 IE 中&#xff0c;我们则需要写&#xff1a;event.returnValue false;jquery &#xff0c;跨浏览器的实现&#xff0c;我们统一只需要写&am…

创建非矩形的Windows 窗体

创建非矩形窗体的过程包含三个步骤&#xff1a;• 创建一个作为窗体图面的位图。&#xff08;一种有效的方式是&#xff0c;您可以从矩形中“裁剪掉”所需的窗体形状。&#xff09;• 创建Windows 应用程序项目&#xff0c;将其属性设置为移除标题栏并使用位图作为窗体背景。•…

ExtJs6 Desktop Demo 修改测试

一直用Extjs4&#xff0c; extjs6 的变化较大&#xff0c;这几天有兴趣研究一下&#xff0c;把自带的Demo的desktop做了些修改&#xff0c; 1.首先下载安装sencha cmd 2.然后 需要生成新项目 用sencha cmd 命令如下&#xff1a; sencha -sdk E:\ext-6.0.0 generate app linb…

包含html语言的超链接标记的网页_零基础入门 HTML 的 8 分钟极简教程

在今天&#xff0c;前端工程师已经成为研发体系中的重要岗位之一。可是与此相对的是&#xff0c;极少大学的计算机专业愿意开设前端课程&#xff0c;大部分前端工程师的知识&#xff0c;也都是在实践和工作中不断学习的。最近收到很多同学的后台留言&#xff0c;说希望多推出一…

mongodb模糊查询_AWS 回击了!推出兼容 MongoDB 的 DocumentDB

2018 年 10 月&#xff0c;MongoDB 将其开源许可证从 GNU AGPLv3 切换到 Server Side Public License(SSPL)&#xff0c;并明确指出之所以会更改开源协议是因为部分云计算公司在使用 MongoDB 的时候没有遵循其开源协议。2019 年 1 月 9 日&#xff0c;AWS 宣布推出 Amazon Docu…

琴岛学院计算机应用技术,我院计算机工程系成功晋级“中国高校计算机大赛-网络技术挑战赛”全国总决赛...

2017年7月22日&#xff0c;由教育部高等学校计算机类专业教学指导委员会、教育部高等学校软件工程专业教学指导委员会、教育部高等学校大学计算机课程教学指导委员会和全国高等学校计算机教育研究会联合主办的“中国高校计算机大赛”首届网络技术挑战赛于济南大学举办。青岛理工…

Android L 的 Tint(着色)

Tint 是什么&#xff1f; Tint 翻译为着色。 着色&#xff0c;着什么色呢&#xff0c;和背景有关&#xff1f;当然是着背景的色。当我们开发 App 的时候&#xff0c;如果使用了 Theme.AppCompat 主题的时候&#xff0c;会发现 ActionBar 或者 Toolbar 及相应的控件的颜色会相应…

《从零开始学Swift》学习笔记(Day 66)——Cocoa Touch设计模式及应用之通知机制...

原创文章&#xff0c;欢迎转载。转载请注明&#xff1a;关东升的博客 通知&#xff08;Notification&#xff09;机制是基于观察者&#xff08;Observer&#xff09;模式也叫发布/订阅&#xff08;Publish/Subscribe&#xff09;模式&#xff0c;是 MVC&#xff08;模型-视图-控…

JUnit3 一次运行多个测试类和进行多次重复测试:使用测试套件和RepeatedTest

测试套件 如果测试类写到很多&#xff0c;每次要进行测试&#xff0c;难道要重新点击每一个测试类来运行&#xff1f;如果有200个测试类要测试呢&#xff1f; 为了解决这个问题&#xff0c;引入了测试套件&#xff08;TestSuite&#xff09;。 通过将多个测试放入套件中&#x…

mysql 不同分区 同时insert_Mysql分区表的原理和优缺点

分区表的原理分区表是由多个相关的底层表实现&#xff0c;这些底层表也是由句柄对象表示&#xff0c;所以我们也可以直接访问各个分区&#xff0c;存储引擎管理分区的各个底层表和管理普通表一样(所有的底层表都必须使用相同的存储引擎)&#xff0c;分区表的索引只是在各个底层…