Linux Qt使用POSIX多线程条件变量、互斥锁(量)

简介: Linux Qt使用POSIX多线程条件变量、互斥锁(量)今天团建,但是文章也要写。酒要喝好,文要写美,方为我辈程序员的全才之路。嘎嘎 之前一直在看POSIX的多线程编程,上个周末结合自己的理解,写了一个基于Qt的用条件变量同步线程的例子。

Linux Qt使用POSIX多线程条件变量、互斥锁(量)
今天团建,但是文章也要写。酒要喝好,文要写美,方为我辈程序员的全才之路。嘎嘎

之前一直在看POSIX的多线程编程,上个周末结合自己的理解,写了一个基于Qt的用条件变量同步线程的例子。故此来和大家一起分享,希望和大家一起交流。

提到线程,如果在UI编程中,总会和一些耗时操作联系在一起。Qt中处理耗时操作通常有两种方式,一种是将耗时操作放在线程中;另一种则是使用QApplication::processEvents(),防止阻塞UI。从更加通用的角度来讲,我是更倾向于线程的,但对于很多初学者来讲,线程还是有一定难度的。比如说需要对线程间共享的数据提供保护,使用互斥量同步、使用条件变量、使用读写锁同步等;各种同步方式用在什么情况下,开始编程时多线程使用的并不多,无法切身体会到这些问题,后来程序写的多了一点儿,慢慢接触到一些多线程的东西,并且自己也可以学习了相关知识,并用到实际程序中。好了,下面以一个实际的例子为背景,来说明Linux POSIX多线程的一些特性。

程序环境:ubuntu 14.04、 Qt 5.5.1、 Posix多线程(C的用法)

这里简单说下我为什么用Linux C的多线程,因为Qt的多编程对于一些线程的终止时含糊不清楚的,并且一个线程被终止后的资源是无法被清理的,所以我选择是相对底层的一些用法,以后有机会我还会添加线程取消和线程退出的操作。

我自己设定的场景是这样的,在UI主线程中通过界面手动向一个线程间共享的队列中push数据,而另外开启的一个线程则一直在while中pop数据,这算是一个变种的生产者和消费者模式吧。

至于条件变量、互斥量(也就是互斥锁)的初始化在这里不再详细说明,只说明一些相对重要的地方。

  1. UI中向队列push数据(生产者生产数)

这是一个槽函数,当在lineEdit中回车后,则会触发该槽函数,由于该队列是线程间的 共享数据,所以使用了互斥锁进行保护,即该槽操作数据的过程中如果有其他线程想要操作数据,则其他线程则会被阻塞,即访问一个已经被加锁的互斥量的线程会被阻塞。

void Widget::on_le_writeNum_returnPressed()
{

int status;

status = pthread_mutex_lock (&mp_processThread->m_structCondition.mutex);
if (status != 0)
    err_abort (status, "Lock mutex");

QString num = ui->le_writeNum->text();
mp_processThread->queuePushData(num.toInt());

status = pthread_cond_signal (&mp_processThread->m_structCondition.cond);

// status = pthread_cond_broadcast( &mp_processThread->m_structCondition.cond);

if (status != 0)
    err_abort (status, "Signal condition");

status = pthread_mutex_unlock (&mp_processThread->m_structCondition.mutex);
if (status != 0)
    err_abort (status, "Unlock mutex");

}

  

  1. 消费者线程pop数据

该线程使用的是Qt的moveToThread方法创建的线程,这里注意的是,整个类都运行在新的线程中。该槽函数随着线程的启动信号(start())发射后而一直进行while循环。首先对互斥量上锁,之后判断谓词状态,如果队列为空,则等待条件变量。等待条件变量时pthread_cond_wait()会自动释放互斥锁,这样其他线程才能够操作共享数据。从条件变量等待中醒来后,会再次获得互斥锁,以操作共享数据。共享数据被操作完成后,再次释放互斥锁。这是我们使用条件变量等待的一个操作流程,如果我们不使用条件变量等待会是怎样的呢?

void ProcessThread::slot_processData()
{

int status;

while(!mb_stopThread)
{
    status = pthread_mutex_lock (&m_structCondition.mutex);
    if (status != 0)
        err_abort (status, "Lock mutex");

    while(m_queue.empty())   //if queue is empty, wait contion
    {

//使用条件变量等待

        status = pthread_cond_wait(&m_structCondition.cond,
                                   &m_structCondition.mutex);

// qDebug() << "pthread_cond_wait is block func!";

        if (status != 0)
        {
            err_abort (status, "Wait on cond faild");
        }
    }

    while(!m_queue.empty())
    {
        qDebug() << "queue mem is" << m_queue.back();

        m_queue.pop();
    }

    status = pthread_mutex_unlock (&m_structCondition.mutex);
    if (status != 0)
        err_abort (status, "Unlock mutex");

}

}

  

  1. 不使用条件变量等待

①不使用条件变量等待

如果不使用条件变量等待,则消费者线程在很大一部时间内几乎都是在执行 while(1)无限循环,这是很占用CPU资源的,在ubuntu下,使用htop查看的效果 如下:

屏蔽status = pthread_cond_wait(&m_structCondition.cond,

&m_structCondition.mutex);

我们看到,此时CPU是满负荷在运行的,这当然不是一个程序所应有的正常状态。

②使用条件变量的结果

此时我们看到CPU的占用率是很低的,这也是为什么使用条件变量的原因之一,让不满足的条件的线程挂起,而不是在浪费CPU资源。条件变量是 允许使用队列的线程之间交换队列状态信息的机制。那么当我们还没有掌握线程条件变量的用法时,又遇到这种情况时,该怎么做呢?简单,加个5ms的延时即可,5ms对我们来讲时间极短极短,但对计算机来讲,已经挺长时间了。

最后,当我们关掉UI窗口时,会有这样一句消息:

QThread: Destroyed while thread is still running

线程正在运行时就被破坏了,这个我们接下来会说,那就是如何退出线程、终止线程以及取消线程等操作了。

欢迎大家一起交流!
原文地址https://www.cnblogs.com/d-h-/p/11286171.html

相关文章
|
安全 Java 编译器
线程安全问题和锁
本文详细介绍了线程的状态及其转换,包括新建、就绪、等待、超时等待、阻塞和终止状态,并通过示例说明了各状态的特点。接着,文章深入探讨了线程安全问题,分析了多线程环境下变量修改引发的数据异常,并通过使用 `synchronized` 关键字和 `volatile` 解决内存可见性问题。最后,文章讲解了锁的概念,包括同步代码块、同步方法以及 `Lock` 接口,并讨论了死锁现象及其产生的原因与解决方案。
215 10
线程安全问题和锁
|
安全 Java 调度
Java编程时多线程操作单核服务器可以不加锁吗?
Java编程时多线程操作单核服务器可以不加锁吗?
148 2
|
9月前
|
并行计算 安全 Java
Python GIL(全局解释器锁)机制对多线程性能影响的深度分析
在Python开发中,GIL(全局解释器锁)一直备受关注。本文基于CPython解释器,探讨GIL的技术本质及其对程序性能的影响。GIL确保同一时刻只有一个线程执行代码,以保护内存管理的安全性,但也限制了多线程并行计算的效率。文章分析了GIL的必要性、局限性,并介绍了多进程、异步编程等替代方案。尽管Python 3.13计划移除GIL,但该特性至少要到2028年才会默认禁用,因此理解GIL仍至关重要。
574 16
Python GIL(全局解释器锁)机制对多线程性能影响的深度分析
|
11月前
|
供应链 安全 NoSQL
PHP 互斥锁:如何确保代码的线程安全?
在多线程和高并发环境中,确保代码段互斥执行至关重要。本文介绍了 PHP 互斥锁库 `wise-locksmith`,它提供多种锁机制(如文件锁、分布式锁等),有效解决线程安全问题,特别适用于电商平台库存管理等场景。通过 Composer 安装后,开发者可以利用该库确保在高并发下数据的一致性和安全性。
153 6
|
10月前
|
Java 关系型数据库 MySQL
【JavaEE“多线程进阶”】——各种“锁”大总结
乐/悲观锁,轻/重量级锁,自旋锁,挂起等待锁,普通互斥锁,读写锁,公不公平锁,可不可重入锁,synchronized加锁三阶段过程,锁消除,锁粗化
|
10月前
|
Java 程序员 调度
【JavaEE】线程创建和终止,Thread类方法,变量捕获(7000字长文)
创建线程的五种方式,Thread常见方法(守护进程.setDaemon() ,isAlive),start和run方法的区别,如何提前终止一个线程,标志位,isinterrupted,变量捕获
|
存储 缓存 安全
【Java面试题汇总】多线程、JUC、锁篇(2023版)
线程和进程的区别、CAS的ABA问题、AQS、哪些地方使用了CAS、怎么保证线程安全、线程同步方式、synchronized的用法及原理、Lock、volatile、线程的六个状态、ThreadLocal、线程通信方式、创建方式、两种创建线程池的方法、线程池设置合适的线程数、线程安全的集合?ConcurrentHashMap、JUC
【Java面试题汇总】多线程、JUC、锁篇(2023版)
|
12月前
|
运维 API 计算机视觉
深度解密协程锁、信号量以及线程锁的实现原理
深度解密协程锁、信号量以及线程锁的实现原理
181 2
|
存储 Java 程序员
优化Java多线程应用:是创建Thread对象直接调用start()方法?还是用个变量调用?
这篇文章探讨了Java中两种创建和启动线程的方法,并分析了它们的区别。作者建议直接调用 `Thread` 对象的 `start()` 方法,而非保持强引用,以避免内存泄漏、简化线程生命周期管理,并减少不必要的线程控制。文章详细解释了这种方法在使用 `ThreadLocal` 时的优势,并提供了代码示例。作者洛小豆,文章来源于稀土掘金。
177 6
|
存储 Ubuntu Linux
C语言 多线程编程(1) 初识线程和条件变量
本文档详细介绍了多线程的概念、相关命令及线程的操作方法。首先解释了线程的定义及其与进程的关系,接着对比了线程与进程的区别。随后介绍了如何在 Linux 系统中使用 `pidstat`、`top` 和 `ps` 命令查看线程信息。文档还探讨了多进程和多线程模式各自的优缺点及适用场景,并详细讲解了如何使用 POSIX 线程库创建、退出、等待和取消线程。此外,还介绍了线程分离的概念和方法,并提供了多个示例代码帮助理解。最后,深入探讨了线程间的通讯机制、互斥锁和条件变量的使用,通过具体示例展示了如何实现生产者与消费者的同步模型。