C++(Qt)软件调试---线程死锁调试(15)

这篇具有很好参考价值的文章主要介绍了C++(Qt)软件调试---线程死锁调试(15)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

C++(Qt)软件调试—线程死锁调试(15)

1、前言

死锁是一种情况,其中两个或多个线程(或进程)相互等待对方释放资源,导致它们都无法继续执行。这是一种非常令人头疼的问题,因为它可以导致程序挂起,无法继续运行。

本文中会详细讲述linux、Windows下调试C++线程死锁、Qt线程死锁的方式。

  • 系统环境:ubuntu20.04、Windows10;
  • 编译器:g++10、MinGW、MSVC2017-64;
  • 调试工具:gdb、WinDbg。
  • 所有程序编译时最好加上调试信息,如果是使用Qt,则使用Debug或者Profile模式。
  • 文中用到的方法也适用于调试死循环,不过细节上有一点点区别。

2、常见死锁

单线程死锁
有时候,线程申请了锁资源,还没有等待释放,又一次申请这把锁,结果就是挂起等待这把锁的释放,但是这把锁是被自己拿着,所以就会永远挂起等待,就造成了死锁。导致重复加锁的原因可能如下:

  • 通常会因为在多分支中加锁,而某个分支忘记了加锁或者因为return、break等语句跳过了锁的释放;
  • 因为程序中自己使用throw抛出异常或者底层库抛出异常,打乱了程序的执行流程,导致锁没有释放。

例如,考虑以下伪代码:

void threadFun1()
{
    g_mutex1.lock();        // 加锁
    
    g_mutex1.lock();        // 重复加锁

    g_mutex1.unlock();
}

void threadFun1()
{
    g_mutex1.lock();        // 加锁
    
    if(value > 10) 
    {
        return;           // 提前返回,跳过释放
    }

    g_mutex1.unlock();
}

void threadFun1()
{
    g_mutex1.lock();        // 加锁
    
    if(value > 10) 
    {
        throw;             // 抛出异常,打乱执行流程,跳过释放
    }

    g_mutex1.unlock();
}

多线程死锁
多线程死锁是更常见的情况,通常在多个线程之间共享资源时发生,也比单线程死锁更难排查。

多线程死锁是指两个或多个线程在等待对方释放资源时被阻塞,无法继续执行。

例如:线程1锁定了lock1并尝试获取lock2,而线程2锁定了lock2并尝试获取lock1,它们彼此等待对方释放资源,从而导致死锁。

/********************************************************************************
* 文件名:   main1.cpp
* 创建时间: 2023-10-25 10:57:54
* 开发者:   MHF
* 邮箱:     1603291350@qq.com
* 功能:     多线程死锁示例
*********************************************************************************/
#include <iostream>
#include <thread>
#include <mutex>
#include <unistd.h>

using namespace std;
mutex mutex1;
mutex mutex2;

void threadA()
{
    cout << "启动线程A" << endl;

    mutex1.lock();
    cout << "线程A上锁mutex1" << endl;

    // 为了模拟死锁,让线程A休眠一段时间
    sleep(1);

    mutex2.lock();                        // 由于线程B已经上锁mutex2,这里会等待线程B解锁
    cout << "线程A上锁mutex2" << endl;

    // 执行一些操作...

    mutex2.unlock();
    mutex1.unlock();
}

void threadB()
{
    cout << "启动线程B" << endl;

    mutex2.lock();
    cout << "线程B上锁 mutex2" << endl;

    // 为了模拟死锁,让线程B休眠一段时间
    sleep(1);

    mutex1.lock();                      // 由于线程A已经上锁mutex1,这里会等待线程A解锁
    cout << "线程B上锁 mutex1" << endl;

    // 执行一些操作...

    mutex1.unlock();
    mutex2.unlock();
}

int main()
{
    thread t1(threadA);
    thread t2(threadB);

    t1.join();
    t2.join();

    return 0;
}

3、linux下gdb调试C++死锁

1.1 使用代码

	/********************************************************************************
* 文件名:   main.cpp
* 创建时间: 2023-10-24 21:40:05
* 开发者:   MHF
* 邮箱:     1603291350@qq.com
* 功能:     单线程死锁示例
*********************************************************************************/
#include<iostream>
#include <thread>
#include <mutex>

using namespace std;

mutex g_mutex1;

void threadFun1()
{
    cout << 1 << endl;
    g_mutex1.lock();        // 加锁
    cout << 2 << endl;
    g_mutex1.lock();        // 重复加锁
    cout << 3 << endl;
}


int main()
{
    thread t1(threadFun1);

    t1.join();
    return 0;
}

1.2 gdb调试

  1. 使用g++ -g main.cpp -lpthread命令编译代码;

  2. 使用./a.out运行程序,会发现程序出现死锁,不会继续执行;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  3. 重新打开一个终端窗口;

  4. 使用ps -aux | grep "a.out\|USER"命令查看a.out程序的进程信息(注意:\| 前后不能有空格);

    • grep “a.out \| USER”:表示只显示包含a.out字符串或者USER字符串的行;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  5. 使用sudo gdb -q -p 14742将gdb附加到a.out的进程PID上(注意附加到进程需要使用sudo);

  6. 进入gdb后使用info threads命令查看所有线程的信息;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  7. 从图中可以看出在线程2的堆栈停止在了**__lll_lock_wait**帧,在这个位置使用了g_mutex1锁,__lll_lock_wait函数是Linux系统中用于实现线程互斥锁等待的函数,它使线程进入等待状态,直到互斥锁可用。

  8. 使用thread 2命令进入到线程2中;

  9. 使用bt命令查看线程2当前的堆栈信息(也可以使用thread apply all bt命令查看所有线程的堆栈);

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  10. 可以堆栈停止在main.cpp文件的第21行,threadFun1()函数中;

  11. 使用f 4命令切换到线程2堆栈的第4帧,可以看见是停止在g_mutex1.lock()这一行加锁的代码上;

  12. 使用list命令查看上下文代码,可以看见加锁了两次;

  13. 使用p g_mutex1命令打印锁的信息可以看见__lock = 2也是加锁了两次。

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

3、linux下gdb调试Qt死锁

1.1 使用代码

#include "widget.h"
#include "ui_widget.h"
#include <QtConcurrent>
#include <QMutex>

QMutex g_mutex;

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
}

Widget::~Widget()
{
    delete ui;
}


void Widget::on_pushButton_clicked()
{
    // 创建一个QtConcurrent线程
    QtConcurrent::run(QThreadPool::globalInstance(), [&]()
    {
        qDebug() << "进入QtConcurrent线程";
        g_mutex.lock();
        qDebug() << "加锁1次";
        g_mutex.lock();
        qDebug() << "加锁2次,重复加锁";

        g_mutex.unlock();
    });
}

1.2 gdb调试

  1. 编译运行Qt程序后,点击pushButton按键,进入QtConcurrent线程,触发死锁;

  2. 使用ps -aux | grep 'testMutex\|USER'命令查看死锁进程pid;

  3. 使用sudo gdb -q -p 21714命令将gdb附加到进程;

  4. 使用info threads命令查看所有线程的信息;

  5. 如下图所示,可看出线程7的类型为Thread(pooled)(如果是使用QThread创建的线程这里类型就是QThread),这是使用线程池创建的QtConcurren线程,停止的堆栈帧的状态为syscall();程序停在syscall()函数通常意味着它正在进行系统调用,而如果出现死锁后线程就会一直处于这种状态;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  6. 使用thread 7命令切换到线程7;

  7. 使用bt命令查看线程7堆栈信息;

  8. 如下图所示,利用看出QBasicMutex::lockInternal()或者QMutex::lock(),表示线程7堆栈停止在互斥锁的lock()函数位置,如何找到包含自己源代码的堆栈帧,在widget.cpp文件的29行。

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  9. 使用f 3命令切换到堆栈的第3帧,可以看的这一帧停止在g_mutex.lock()位置,正在加锁位置;

  10. 使用list命令查看上下文代码,可以看出加锁两次;

  11. 使用p g_mutex命令打印g_mutex锁的信息,和c++中的mutex锁不同,QMutex锁打印无法获得有帮助的信息。

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

4、Windows下gdb调试C++死锁

使用代码和linux下一样。

  1. 打开MinGW-64的cmd窗口(从这里打开具有完整的环境变量,便于找到依赖库);

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 进入到源代码所在路径;

  2. 使用g++.exe main.cpp -g -lpthread命令编译代码(如果提升找不到g++则使用MinGw所在绝对路径);

  3. 执行a.exe程序,触发死锁;

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 打开任务管理器,找到a.exe程序,右键选择【转到详细信息】,查看进程的pid号,

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 再打开一个cmd窗口;

  2. 使用gdb -q -p 8740将gdb附加到进程调试;

  3. 使用info threads命令查看所有线程信息(和linux下不同,不能直接看出死锁线程);

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 使用thread apply all bt查看所有线程的堆栈信息;

  2. 如下图所示可以看出在线程2中出现了pthread_mutex_lock(),表示这个线程的堆栈停止在上锁位置,所以出现死锁,再往下找发现死锁位置出现在main.cpp文件的第21行中,threadFun1()函数位置。

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  3. 后面操作就可有可无了,并且和linux下没有什么区别;

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

5、Windows下gdb调试Qt死锁

使用代码和linux下的相同;

注意:Windows下使用MinGW编译程序,调试时选择的gdb版本应该和编译的g++版本相同,不能使用32位的gdb调试64位的程序,或者相反。

  1. Qt编译运行程序后,触发死锁;

  2. 打开对应版本的MinGW的cmd终端;

  3. 使用任务管理器窗口死锁程序的pid进程号;

  4. 使用gdb -q -p pid将gdb附加到死锁进程;

  5. 直接使用thread apply all bt显示所有线程的堆栈信息;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  6. 可以看出线程3出现死锁,后续操作都是一样的。

  7. 不过MinGW中gdb调试有时会出现下列情况,无法进行调试,目前没找到问题;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

6、Windows下Windbg调试C++死锁

1.1 使用代码

  • 直接使用C++中的mutex锁重复上锁在msvc编译器中会在触发时抛出异常,所以无需调试。
  • 这里改为使用多线程死锁进行演示。
/********************************************************************************
* 文件名:   main.cpp
* 创建时间: 2023-10-25 10:57:54
* 开发者:   MHF
* 邮箱:     1603291350@qq.com
* 功能:     多线程死锁示例
*********************************************************************************/
#include <iostream>
#include <thread>
#include <mutex>
#include <Windows.h>

using namespace std;
mutex mutex1;
mutex mutex2;

void threadA()
{
    cout << "start A" << endl;

    mutex1.lock();
    cout << "threadA mutex1 lock" << endl;

    // 为了模拟死锁,让线程A休眠一段时间
    Sleep(1000);

    mutex2.lock();                        // 由于线程B已经上锁mutex2,这里会等待线程B解锁
    cout << "threadA mutex2 lock" << endl;

    // 执行一些操作...

    mutex2.unlock();
    mutex1.unlock();
}

void threadB()
{
    cout << "start B" << endl;

    mutex2.lock();
    cout << "threadB mutex2 lock" << endl;

    // 为了模拟死锁,让线程B休眠一段时间
    Sleep(1000);

    mutex1.lock();                      // 由于线程A已经上锁mutex1,这里会等待线程A解锁
    cout << "threadB mutex1 lock" << endl;

    // 执行一些操作...

    mutex1.unlock();
    mutex2.unlock();
}

int main()
{
    thread t1(threadA);
    thread t2(threadB);

    t1.join();
    t2.join();

    return 0;
}

1.2 Windbg调试

  1. 使用MSVC编译器编译代码,运行并触发死锁;

  2. 打开WinDbg程序,(在C:\Program Files\Windows Kits\10\Debuggers\x64路径下);

  3. 选择【File】->【Attach to Process】或者直接按快捷键F6;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  4. 然后选择By ID,找到死锁进程,然后点击【OK】;

    C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  5. 然后输入~*k命令查看所有线程的堆栈信息,如下所示出现std::_Mutex_base::lock字样,可看出在线程1、2出现死锁;

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 然后选择【View】,打开【Processes and Threads】窗口和【Calls Stack】窗口;

  2. 点击【Processes and Threads】窗口中的线程1,再点击【Calls Stack】窗口中的堆栈帧,就可以跳转到出现死锁的源码位置;

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 或者直接点击Command窗口中的堆栈帧也可以跳转到死锁源码位置(不过在WinDbg中定位到源码的位置是实际位置的下一行)。

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

7、Windows下Windbg调试Qt死锁

使用代码和Linux下的相同;

  1. 前面步骤都是相同的;

  2. 在使用~*k命令窗口所有线程的堆栈信息时会发现看不到太多有帮助的信息,这时可用找包含源码文件的堆栈帧;

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 如图所示,点击这一帧就可以跳转到源码查看是否时出现死锁的位置;

C++(Qt)软件调试---线程死锁调试(15),# C++软件调试,c++,qt,GDB,Windbg,死锁

  1. 如果想要查看更加详细的调试信息,需要到Qt官网下载Qt库的调试符号。

{__/}
(̷ ̷´̷ ̷^̷ ̷`̷)̷◞~❤
| ⫘ |文章来源地址https://www.toymoban.com/news/detail-719725.html

到了这里,关于C++(Qt)软件调试---线程死锁调试(15)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C++(Qt)软件调试---静态分析工具clang-tidy(18)

    更多精彩内容 👉个人内容分类汇总 👈 👉C++软件调试、异常定位 👈 现在很多人在开发中完全忽略了编译器、IDE的警告提示,这怎么可能写出稳定的程序。 clang-tidy是一个由LLVM项目提供的开源工具,是一个静态分析工具,用于进行静态代码分析和代码质量改进。 支持C++/C

    2024年01月23日
    浏览(37)
  • QT5.15.2搭建Android编译环境及使用模拟器调试(全)

    地址:下载 我电脑的windows的,所以选windows 由于官方安装过程非常非常慢,一定要跟着步骤来安装,不然慢到怀疑人生 1)打开\\\"命令提示符\\\"(开始 - Windows 系统 - 命令提示符) 或者 “win+R” -输入cmd 找到刚才的exe 命令行输入以下命令(exe位置根据自己实际的),然后回车

    2024年02月04日
    浏览(55)
  • 音视频源码调试前准备vs2019+qt5.15.2搭建可调试环境

    安装vs2019+qt,并且在windows环境上安装ffmpeg,尝试使用qt+cdb进行调试,尝试使用vs2019加载qt的程序。 安装VS2019+5.12.2qt环境,并进行测试。 1:安装Visual Studio 2019, a.从官网下载,或者vs2019社区版本下载地址 https://www.jb51.net/softs/700418.htm ====》 安装时必须勾选 Desktop development with C++

    2024年02月05日
    浏览(56)
  • 15-1_Qt 5.9 C++开发指南_Qt多媒体模块概述

    多媒体功能指的主要是计算机的音频和视频的输入、输出、显示和播放等功能,Qt 的多媒体模块为音频和视频播放、录音、摄像头拍照和录像等提供支持,甚至还提供数字收音机的支持。本章将介绍 Qt 多媒体模块的功能和使用。 Qt 多媒体模块提供了很多类,可以实现如下的

    2024年02月13日
    浏览(50)
  • QT——C++ 多线程05

    QT创建(使用)多线程的方式有三种。 直接创建QThread 对象,重写run方法,最后调用start方法启动线程。 通过调用QObject类提供的moveToThread方法实现 线程池 如果需要向run传入参数,或者返回执行结果,可以通过信号和槽机制实现 mywork.h mywork.cpp mainwindow.h mainwindow.cpp 更改此对象

    2024年02月11日
    浏览(30)
  • 线程池及gdb调试多线程

    概念:通俗的讲就是一个线程的池子,可以循环的完成任务的一组线程集合 必要性: 我们平时创建一个线程,完成某一个任务,等待线程的退出。但当需要创建大量的线程时,假设 T1 为 创建线程时间, T2 为在线程任务执行时间,T3 为 线程销毁时间 , 当  T1+T3 T2 ,这时候

    2024年02月07日
    浏览(44)
  • Qt6 c++教程9测试&调试

    调试和测试是软件开发的重要组成部分。在本章中,你将学习如何调试 Qt 项目、不同的调试技术以及 Qt 支持的调试器。调试是发现错误或不希望出现的行为的根本原因并加以解决的过程。我们还将讨论使用Qt Test框架进行单元测试。Qt Test是基于Qt的应用程序和库的单元测试框

    2024年02月05日
    浏览(41)
  • 6.10 线程池及gdb调试多线程

    线程池概念和使用 概念: 通俗的讲就是一个线程的池子,可以循环的完成任务的一组线程集合 必要性: 我们平时创建一个线程,完成某一个任务,等待线程的退出。但当需要创建大量的线程时,假设T1为创建线程时间,T2为在线程任务执行时间,T3为线程销毁时间,当 T1+

    2024年02月08日
    浏览(45)
  • 1.13|1.14|1.15|1.6、GDB调试

    输入这两条命令,第一条用于 生成 调式代码,第二条 运行 调试代码 test.c 需要与被调试的代码 test 在 同一个目录 下 list 默认显示 main 函数 ①用list查看代码 list 行号 ,显示 行号上下文 代码 list 函数名 ,表示显示 函数名上下文 的代码 list 文件名:行号 查看非当前文件的代

    2023年04月23日
    浏览(73)
  • 【C++ QT项目2】——高仿安信可串口调试助手

      串口调试助手是一种串口通讯测试工具,它可以用于打开、关闭、配置串口,读写串口数据等常见的串口通信操作。 在嵌入式系统调试、模块测试、通讯协议分析等领域都具有广泛的应用。   串口助手通常提供GUI界面,让用户可以更加方便、直观地进行串口通讯测试

    2024年02月19日
    浏览(50)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包