Qt多线程之死锁

article/2025/10/27 16:43:31

目录

一、造成死锁的原因

现象分析

二、死锁的概念和发生条件

三、死锁的避免

四、避免死锁的示例代码


一、造成死锁的原因

一般性原则,每个临界资源都需要一个线程锁来进行保护,假设有多个线程锁该怎么设计程序呢?先来看以下的一个示例(代码很简单就不加注释了):

#include <QCoreApplication>
#include <QThread>
#include <QMutex>
#include <QDebug>int g_i_current_data = 0;
QMutex g_mutex_1;
QMutex g_mutex_2;class ThreadMutexA : public QThread
{
protected:void run(){qDebug() << objectName() << ": run begin...";while( true ){g_mutex_1.lock();qDebug() << objectName() << "get mutex_1...";g_mutex_2.lock();qDebug() << objectName() << "get mutex_2...";g_i_current_data++;qDebug() << objectName() << ": " << g_i_current_data;g_mutex_2.unlock();g_mutex_1.unlock();msleep(1);}qDebug() << objectName() << ": run end...";}
};class ThreadMutexB : public QThread
{
protected:void run(){qDebug() << objectName() << ": run begin...";while( true ){g_mutex_2.lock();qDebug() << objectName() << "get mutex_2...";g_mutex_1.lock();qDebug() << objectName() << "get mutex_1...";g_i_current_data--;qDebug() << objectName() << ": " << g_i_current_data;g_mutex_1.unlock();g_mutex_2.unlock();msleep(1);}qDebug() << objectName() << ": run end...";}
};int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);qDebug() << "main begin...";ThreadMutexA thread_a;ThreadMutexA thread_a1;ThreadMutexA thread_a2;ThreadMutexB thread_b;ThreadMutexB thread_b1;thread_a.setObjectName("ThreadMutexA");thread_a1.setObjectName("ThreadMutexA_1");thread_a2.setObjectName("ThreadMutexA_2");thread_b.setObjectName("ThreadMutexB");thread_b1.setObjectName("ThreadMutexB_1");thread_a.start();thread_a1.start();thread_a2.start();thread_b.start();thread_b1.start();qDebug() << "main end...";return a.exec();
}

现象分析

1、现象:五个线程分别都进入了run()函数中,线程锁1和线程锁2分别被线程A和线程B获取(这两个线程动作比较快优先获取到线程数)后,程序就阻塞在这里了,不继续往下执行了。

2、分析:

(1)当线程A获取线程锁1时,就要继续获取线程锁2,但是此时线程锁2被线程B获取,线程A就只好等待线程锁2被释放;

(2)线程B获取线程锁2后,需要继续获取线程锁1,而此时线程锁1被线程A锁获取并没有释放,线程B也就只好等待线程A释放线程锁1了;

(3)两个线程一直等待彼此释放相应线程锁,等到花都谢了也等不到,程序无法继续执行。

3、原因:造成了线程的死锁

二、死锁的概念和发生条件

1、死锁概念:多线程之间相互等待临界资源而造成彼此无法继续执行。

2、死锁发生条件:(1)系统中存在多个临界资源且临界资源不可抢占;(2)每个线程需要多个临界资源才能继续执行。

三、死锁的避免

1、对所有临界资源都分配唯一的序号(R1, R2, R3, ..., Rn);

2、对应的线程锁也分配同样的序号(M1, M2, M3, ..., Mn);

3、系统中的每个线程必须按照严格的升序的次序去请求资源。

    ThreadMutexA A;mutex_1.lock();mutex_2.lock();mutex_3.lock();mutex_4.lock();

四、避免死锁的示例代码

#include <QCoreApplication>
#include <QThread>
#include <QMutex>
#include <QDebug>int g_i_current_data = 0;
QMutex g_mutex_1;
QMutex g_mutex_2;class ThreadMutexA : public QThread
{
protected:void run(){qDebug() << objectName() << ": run begin...";while( true ){g_mutex_1.lock();qDebug() << objectName() << "get mutex_1...";g_mutex_2.lock();qDebug() << objectName() << "get mutex_2...";g_i_current_data++;qDebug() << objectName() << ": " << g_i_current_data;g_mutex_2.unlock();g_mutex_1.unlock();msleep(1);}qDebug() << objectName() << ": run end...";}
};class ThreadMutexB : public QThread
{
protected:void run(){qDebug() << objectName() << ": run begin...";while( true ){g_mutex_1.lock();qDebug() << objectName() << "get mutex_1...";g_mutex_2.lock();qDebug() << objectName() << "get mutex_2...";g_i_current_data--;qDebug() << objectName() << ": " << g_i_current_data;g_mutex_2.unlock();g_mutex_1.unlock();msleep(1);}qDebug() << objectName() << ": run end...";}
};int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);qDebug() << "main begin...";ThreadMutexA thread_a;ThreadMutexA thread_a1;ThreadMutexA thread_a2;ThreadMutexB thread_b;ThreadMutexB thread_b1;thread_a.setObjectName("ThreadMutexA");thread_a1.setObjectName("ThreadMutexA_1");thread_a2.setObjectName("ThreadMutexA_2");thread_b.setObjectName("ThreadMutexB");thread_b1.setObjectName("ThreadMutexB_1");thread_a.start();thread_a1.start();thread_a2.start();thread_b.start();thread_b1.start();qDebug() << "main end...";return a.exec();
}


http://chatgpt.dhexx.cn/article/a3OOB5BW.shtml

相关文章

QT多线程网络通信

QT多线程网络通信 使用QT提供的类进行基于TCP的套接字通信需要用到两个类&#xff1a; 1、QTcpServer 服务器类&#xff0c;用于监听客户端连接以及和客户端建立连接 2、QTcpSocket 通信的套接字类&#xff0c;客户端、服务器端都需要使用 ​ 这两个套接字通信类都属于网络…

QT 多线程应用

QT多线程的实现有两种方法&#xff0c;一种是继承QThread的多线程使用方法&#xff0c;另外一种是使用QObject实现多线的方法。传统的方式是继承QTread&#xff0c;但是这种方式比较的容易出错&#xff0c;QT官方推荐使用的是第二种方式。这里介绍这两种方式的最简单的一种创建…

Qt 多线程(QThread)

Qt 多线程&#xff08;QThread&#xff09; 一.理解QThread 关于QThread&#xff0c;参考qt助手中的解析 Detailed Description A QThread object manages one thread of control within the program. QThreads begin executing in run(). By default, run() starts the eve…

QT 多线程的实现方法以及GUI线程与其他线程间的通信

GUI线程 Qt应用程序exec后就会生成一个线程&#xff0c;这个线程就是主线程&#xff0c;在GUI程序中也称为GUI线程。主线程也是唯一允许创建QApplication或QCoreAppliation对象&#xff0c;比并且可以对创建的对象调用exec()的线程&#xff0c;从而进入事件循环。 在只有主线…

QT多线程接收串口数据

** QT多线程接收串口数据 ** 1.前言 QT多线程的使用&#xff0c;和绝大数人一样&#xff0c;犯了错误&#xff08;请查阅Qt开发人员( Bradley T. Hughes)Blog中的文章 you are-doing-it-wrong介绍&#xff09;。为了解决问题&#xff0c;网上查阅学习了几十篇文章&#xff0c…

Qt多线程互斥

目录 一、多线程与临界资源的依赖 现象分析 二、互斥和解决方法 三、QMutex的主要成员函数和使用 四、示例代码 五、小结 一、多线程与临界资源的依赖 除了上一节所说的&#xff0c;多线程在代码执行的时序上会有依赖&#xff0c;那么其他地方是否还有所依赖呢&#xff…

QT 多线程创建方法及应用实例

QT 多线程创建方法及应用实例 方法一&#xff1a; &#xff08;1&#xff09;创建一个QT应用 &#xff08;2&#xff09;创建线程类&#xff0c;继承QThread simplethreadone.h #ifndef SIMPLETHREADONE_H #define SIMPLETHREADONE_H #include <QThread>class SimpleThr…

Qt 多线程专栏

主线程就是进程&#xff0c;进程代表4个G的资源空间&#xff0c;主线程退出就意味着进程退出&#xff0c;在Qt中&#xff0c;主线程不能是子线程的父对象&#xff0c;因此必须在线程的析构函数中去delete线程对象。 只要定时器启动&#xff0c;自动触发timeout()信号 类A继承了…

Qt多线程编程

由于疫情原因被封控在家&#xff0c;相信大家也有跟我类似的情况&#xff0c;在学习知识提升和充实自己&#xff0c;下面我将这几天在学习的Qt多线程编程做个简单的总结&#xff0c;随此笔记的还有几个简单的例子&#xff0c;以便后续开发过程中回忆和参考。 Qt的多线程开发&am…

Qt多线程

Qt多线程有两种方法&#xff1a; 1、定义一个QThread的派生类&#xff0c;重写run函数&#xff0c;run函数中的程序就是新线程中要执行的内容。在主线程中实例化该类&#xff0c;调用start函数&#xff0c;就可以实现多线程。 2、定义一个QObject的派生类&#xff0c;把要在新线…

Qt创建多线程的两种方法

来源&#xff1a;https://github.com/czyt1988/czyBlog/tree/master/tech/QtThread 1.摘要 Qt有两种多线程的方法&#xff0c;其中一种是继承QThread的run函数&#xff0c;另外一种是把一个继承于QObject的类转移到一个Thread里。 Qt4.8之前都是使用继承QThread的run这种方法…

Qt----多线程

文章目录 多线程一、继承QThread的线程1. 应用实例2. 程序运行效果 二、继承QObject的线程1. 应用实例2. 程序运行效果 多线程 我们写的一个应用程序&#xff0c;应用程序跑起来后一般情况下只有一个线程&#xff0c;但是可能也有特殊情况。比如我们前面章节写的例程都跑起来后…

Qt5教程(九):Qt多线程

目录&#xff1a; 一、创建工程 二、QThread 源码一览 三、QThread相关方法介绍 四、创建线程 一、创建工程 先创建一个工程吧, 具体步骤前面讲过很多次了&#xff0c; 就不再细说了。 然后在Header文件夹下创建添加一个头文件, 右键Headers -> Add New... -> C -&g…

Qt 多线程的几种实现方式

Qt多线程的实现方式有&#xff1a; 1. 继承QThread类&#xff0c;重写run()方法 2. 使用moveToThread将一个继承QObject的子类移至线程&#xff0c;内部槽函数均在线程中执行 3. 使用QThreadPool,搭配QRunnable&#xff08;线程池&#xff09; 4. 使用QtConcurrent&#xf…

QT中的多线程

目录 1、QThread介绍 1.2、继承Qthread类 1.2.1、得到线程id 1.2.2、让线程一直执行 1.2.3、线程退出 1.2.4、​​​​​​​connect的第五个参数 1.2.5、线程锁 QMutex ​​​​​​​ 1、QThread介绍 QThread类提供了一个与平台无关的管理线程的方法。一个QThread对…

faild to create process解决办法

win7下运行pip时报faild to create process的解决办法&#xff1a; 1.找到python.exe文件&#xff0c;复制其文件路径。如下图&#xff0c;python.exe路径为D:\interpreter\python.exe。 2.找到pip-script.py文件&#xff0c;打开。将步骤1中复制的路径粘贴在第一行&#xff0…

CreateProcess error=2, 系统找不到指定的文件 解决方法

CreateProcess error2, 系统找不到指定的文件 解决方法 一般这是由于ndk缺少文件引起的 解决方法1&#xff1a; 在项目根目录下的local.properties文件中加cmd后缀 方法二下载16b的版本替换原来的ndk-bundle目录&#xff0c;默认在C:\Users\用户名\AppData\Local\Android\S…

SQL SERVER 2008 执行xp_cmdshell的过程中出错,调用createprocess失败,错误代码 5 解决方案

1、进入 控制面板----管理工具------本地安全策略 点击 本地策略--------安全选项 把“网络安全:LAN管理器身份验证级别”双击打开&#xff0c;选择“发送 LM 和 NTLM 响应”&#xff08;原来是“没有定义”&#xff09;。确认后&#xff0c;重新启动服务器&#xff0c;该问题…

runnerw.exe: CreateProcess failed with error 5:

在idea配置Git时遇到这个问题&#xff1a; 后来发现是我路径设置错了&#xff1a; 解决方案 修改设置里的路径即可&#xff08;Setings-Version Control-Git&#xff09;

Qt creater出现“启动程序失败,路径或者权限错误”或“The process could not be started!”解决方法

Qt creater出现“启动程序失败&#xff0c;路径或者权限错误”或“The process could not be started&#xff01;”解决方法 错误示例 远古版本的QTCreator在编译程序时没问题&#xff0c;在运行程序时会提示“The process could not be started&#xff01;" 较新版本…