QThread使用_qthread requestinterruption-程序员宅基地

技术标签: Qt  


QThread使用的两种方式

  Qt提供了两种线程的使用方式,分别如下:

moveToThread

 class Worker : public QObject
  {
    
      Q_OBJECT

  public slots:
      void doWork(const QString &parameter) {
    
          QString result;
          /* ... here is the expensive or blocking operation ... */
          emit resultReady(result);
      }

  signals:
      void resultReady(const QString &result);
  };

  class Controller : public QObject
  {
    
      Q_OBJECT
      QThread workerThread;
  public:
      Controller() {
    
          Worker *worker = new Worker;
          worker->moveToThread(&workerThread);
          connect(&workerThread, &QThread::finished, worker, &QObject::deleteLater);
          connect(this, &Controller::operate, worker, &Worker::doWork);
          connect(worker, &Worker::resultReady, this, &Controller::handleResults);
          workerThread.start();
      }
      ~Controller() {
    
          workerThread.quit();
          workerThread.wait();
      }
  public slots:
      void handleResults(const QString &);
  signals:
      void operate(const QString &);
  };

  这种方式下worker的slot将会执行在一个独立的线程,但是,我们仍然能够自由的使用connect通过queue connection,这是线程安全的。

subclass QThread


  class WorkerThread : public QThread
  {
    
      Q_OBJECT
      void run() override {
    
          QString result;
          /* ... here is the expensive or blocking operation ... */
          emit resultReady(result);
      }
  signals:
      void resultReady(const QString &s);
  };

  void MyObject::startWorkInAThread()
  {
    
      WorkerThread *workerThread = new WorkerThread(this);
      connect(workerThread, &WorkerThread::resultReady, this, &MyObject::handleResults);
      connect(workerThread, &WorkerThread::finished, workerThread, &QObject::deleteLater);
      workerThread->start();
  }

  这种方式需要注意:QThread的实例化版本存在于创建其的线程中,而不是新的线程,对于这种方式而言,只有run函数是运行在一个独立的线程中,所以所有QThread的queue slots都将在老的线程中执行,这意味这如果我们连接槽,那么就要考虑临界资源问题。

Qthread同步

QMutex

  QMutex无疑是最常用的线程同步方式,其主要作用是:如果一个线程已经获得了临界资源,第二个线程访问时就会被挂起,直到第一个线程解锁。我们可以像下面这样来使用QMutex:

QMutex mutex;
int number = 6;

void method1()
{
    
  mutex.lock();
  number *= 5;
  number /= 4;
  mutex.unlock();
}

void method2()
{
    
  mutex.lock();
  number *= 3;
  number /= 2;
  mutex.unlock();
}

  这种方式本身并没有什么问题,但是可能出现这样一种情况,即在unlock()之前函数就返回了,这是一种很常见的情况,那么此时程序就会被冻结;为了避免这种情况,所以Qt提供了QMutexLocker,它们锁定一个资源,然后在销毁时释放资源。我们可以像下面这样使用:

QMutex mutex;
int complexFunction(int flag)
{
    
	QMutexLocker locker(&mutex);
	...
	return 1;
}

  与QMutex类似的还有QReadWriteLock,其与QMutex的不同在于:QMutex将会锁定资源,包括读写,而QReadWriteLock则支持多个线程同时读取,但是只能一个线程进行写入。从效率上来说,QReadWriteLock更胜一筹。QReadWriteLock也提供了QReadLocker and QWriteLocker以简化使用。

QWaitCondition

  QWaitCondition契合了多线程中的生产者 - 消费者模式,考虑一下如果只使用QMutex进行生产者与消费者之间的线程同步会是怎样的情形:当生产者生产产品时未满时,消费者无法消费;当产品还未消费完时,生产者不能生产。这种方式将会带来严重的效率问题,所以这种情形下就需要更加灵活的线程同步,因此需要QWaitCondition。示例代码如下:

  const int DataSize = 100000;

  const int BufferSize = 8192;
  char buffer[BufferSize];

  QWaitCondition bufferNotEmpty;
  QWaitCondition bufferNotFull;
  QMutex mutex;
  int numUsedBytes = 0;

class Producer : public QThread
  {
    
  public:
      Producer(QObject *parent = NULL) : QThread(parent)
      {
    
      }

      void run() override
      {
    
          for (int i = 0; i < DataSize; ++i) {
    
              mutex.lock();
              if (numUsedBytes == BufferSize)
                  bufferNotFull.wait(&mutex);
              mutex.unlock();

              buffer[i % BufferSize] = "ACGT"[QRandomGenerator::global()->bounded(4)];

              mutex.lock();
              ++numUsedBytes;
              bufferNotEmpty.wakeAll();
              mutex.unlock();
          }
      }
  };

 class Consumer : public QThread
  {
    
      Q_OBJECT
  public:
      Consumer(QObject *parent = NULL) : QThread(parent)
      {
    
      }

      void run() override
      {
    
          for (int i = 0; i < DataSize; ++i) {
    
              mutex.lock();
              if (numUsedBytes == 0)
                  bufferNotEmpty.wait(&mutex);
              mutex.unlock();

              fprintf(stderr, "%c", buffer[i % BufferSize]);

              mutex.lock();
              --numUsedBytes;
              bufferNotFull.wakeAll();
              mutex.unlock();
          }
          fprintf(stderr, "\n");
      }

  signals:
      void stringConsumed(const QString &text);
  };

解析:

  • 生产者使用对临界资源使用mutex.lock()上锁,然后检查缓冲区是否满了,如果满了,那么bufferNotFull进入等待,此时可以认为mutex暂时释放,生者者将阻塞在这里。
  • 消费者消费产品,产品不再满仓,那么bufferNotFull.wakeAll(),条件激活,但由于消费者中mutex.lock(),生产者暂不启动,但一旦mutex.unlock(),生产者立即获得mutex的控制权,直到mutex.unlock();

  这里需要注意的一点是:为什么使用了QWaitCondition还要使用QMutex呢?对此,Qt assistant中给出了解释:

QWaitCondition allows a thread to tell other threads that some sort of condition has been met. One or many threads can block waiting for a QWaitCondition to set a condition with wakeOne() or wakeAll(). Use wakeOne() to wake one randomly selected thread or wakeAll() to wake them all.

  也就是说:也就是说,线程获得互斥锁,但是会因为QWaitConation不满足而阻塞,直到有使用wakeOne或者wakeAll来激活。
  关于QWaitCondition,有一篇很好的博客可以帮助我们理解,链接如下:https://blog.csdn.net/flyoxs/article/details/54617342

总结:以上只介绍了QMutex、QWaitCondition,但是并不意味着线程同步方式只有这两种,只是因为它们具有代表性而已,至于其他方式,我们还可以使用QSemaphore,这里就不再详述,有兴趣的同学可以去研究一下。

优雅的取消线程

  怎样结束一个long Task呢?比较常用的方式如下:

while(!bStop)
{
    
...
}

  通过控制bStop的状态,我们可以结束一个长时间的任务,但是这种使用方式需要注意:因为bStop是一个临界资源,当我们改变它的状态时,需要考虑互斥问题。但实际上,Qt已经为我们做了这方面的考虑,主要依赖于以下两个函数:

  • void QThread::requestInterruption()
  • bool QThread::isInterruptionRequested() const

  前者只是发起一个中断请求,是否响应完全由线程本身决定。后者则对应前者:如果已经发起中断请求,返回true,否则为false。这两个函数都是线程安全的,其源码如下:

void QThread::requestInterruption()
{
    
    Q_D(QThread);
    QMutexLocker locker(&d->mutex);
    if (!d->running || d->finished || d->isInFinish)
        return;
    if (this == QCoreApplicationPrivate::theMainThread) {
    
        qWarning("QThread::requestInterruption has no effect on the main thread");
        return;
    }
    d->interruptionRequested = true;
}
 
bool QThread::isInterruptionRequested() const
{
    
    Q_D(const QThread);
    QMutexLocker locker(&d->mutex);
    if (!d->running || d->finished || d->isInFinish)
        return false;
    return d->interruptionRequested;
}

  从源码中可以看出,Qt实际已经在实现中加了锁。因此,我们可以放心使用这种方式如下所示:

 void long_task() {
    
       forever {
    
          if ( QThread::currentThread()->isInterruptionRequested() ) {
    
              return;
          }
      }
  }

  需要注意的是,在Qt助手中,建议我们不要调用isInterruptionRequested()过于频繁,以保持cpu的低开销。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/wdl20170204/article/details/111050801

智能推荐

Springboot注解与分析之:@ComponentScan注解_@componentscan 和 context:component-scan base-packa-程序员宅基地

文章浏览阅读219次。Springboot注解与分析之:@ComponentScan注解)使用Spring框架开发Java Web项目的读者一定都用过@Controller、@Service、@Repository等注解。查看源码会发现,这些注解上都会标注一个共同的注解@Component。而在Spring IOC容器中@Controller、@Service、@Repository、@Component等注解的默认装配标识是@ComponentScan注解。<!-- 自动扫描 --><context:_@componentscan 和 context:component-scan base-package 优先级

机器学习实战第三章(决策树)_mydat,labels-程序员宅基地

文章浏览阅读968次。第二章介绍的k-近邻算法可以完成很多分类任务,但是最大缺点是无法给出数据的内在含义,决策树的主要优势就在于数据形式非常容易理解。决策树: 优点:计算复杂度不高,输出结果易于理解,对中间值的缺失不敏感,可以处理不相关特征数据 缺点:可能会产生过度匹配问题 树用数据类型:数值型和标称型。在构造决策树时,我们需要解决的第一个问题就是,当前数据集上哪个特征在划分数据分类时起决_mydat,labels

LR(0)分析表的构建_lr分析表-程序员宅基地

文章浏览阅读2.6w次,点赞58次,收藏300次。LR(0)分析表的构建_lr分析表

【毕业设计】基于单片机的火灾报警系统 -stm32 单片机物联网-程序员宅基地

文章浏览阅读4.4k次,点赞5次,收藏75次。Hi,大家好,这里是丹成学长,今天向大家介绍一个学长做的单片机项目大家可用于 课程设计 或 毕业设计单片机-嵌入式毕设选题大全及项目分享:https://blog.csdn.net/m0_71572576/article/details/1254090521、MCU采用stm32;2、使用MQ2传感器采集烟雾等有害气体,当检测到有害气体时发声报警,并可以邮件通知用户;3、使用DS18B20传感器检测环境温度和火焰;4、采用OLED12864显示屏进行数据显示;5、可设置传感器阈值,修改检测灵敏度;正常状态加_基于单片机的火灾报警系统

okhttp3 springboot post 接收与传参_java okhttp3接收请求体-程序员宅基地

文章浏览阅读4.3k次。springboot+android 收参 传参_java okhttp3接收请求体

ZZULIOJ 1188: 选票统计(一)(结构体专题)_c++题目描述 某单位进行选举,有5位候选人:zhang、wang、zhao、liu、miao。编写-程序员宅基地

文章浏览阅读1.2k次,点赞3次,收藏4次。zzulioj 1188: 选票统计(一)(结构体专题)题目描述某单位进行选举,有5位候选人:zhang、wang、zhao、liu、miao。编写程序,统计每人所得的票数。要求每人的信息里包括两部分:name和votes,分别描述姓名和所得票数。每个人的信息用一个结构体来表示,5个人的信息使用结构体数组。输入首先输入一个整数n,表示一张选票,接下来n行,每行是一个由小写英文字母组成的字符串,表示以上5个候选人之一。输出输出5行,按zhang、wang、zhao、liu、miao的顺序输出5个候_c++题目描述 某单位进行选举,有5位候选人:zhang、wang、zhao、liu、miao。编写程

随便推点

Ext2.0教程下载_ppt版-程序员宅基地

文章浏览阅读114次。 下载地址:http://download.csdn.net/source/594644 自己写的公司培训ppt教程。里面有三个完整的例子源码,可以直接打开运行,附加详细注释。详细介绍见: Ext2.X教程一:Ext介绍以及 ext页面布局 链接:http://blog.csdn.net/wayfoon322/archive/2008/08/27/2836057.aspx Ex..._ext2.0中文下载

[GN] 配置armcc工具链_armcc v5 toolchain-程序员宅基地

文章浏览阅读2.5k次。对于gn来说,工具链被抽象为一系列的工具描述,我们要做的就是写对应的工具描述文件出来。首先在工程根目录建立一个文件.gn(注意文件的名字为空,后缀为.gn),文件中配置工具链描述文件所在的路径buildconfig = "//build/buildconfig.gn"该变量定义了构建工具链描述文件所在的路径,//表示工程根目录。在buildconfig.gn中定义工具链。s..._armcc v5 toolchain

Cisco AP 无法进入全局模式-程序员宅基地

文章浏览阅读351次,点赞9次,收藏6次。用configure t 无法进入全局模式!配置不了 cisco AIR-AP2802I-H-K9。

react-router-dom6,react路由6和5的区别,一文搞懂react-router-dom6-程序员宅基地

文章浏览阅读1.3k次,点赞3次,收藏6次。router-router-dom6 就是当前react路由的最新版本 在2021年11月发布 已经是路由推荐版本。_react-router-dom6

android 标题字体大小,android setTitle怎么实现字体的大小-程序员宅基地

文章浏览阅读279次。烧仙草VB主题里添加如下,可以修改title的字体的大小和样式比如没有ActionBar的Activity中,定义如下的样式false@style/WindowTitle#fff20spbold如果是在ActionBar存在的前提下,主题样式要如下@style/TextAppearance@color/main_color@style/ActionBar@style/TextAppearance..._安卓settitle 设置字体大小

[Unity基础]打包与读取AssetBundle_解包出来的prefab怎么读取-程序员宅基地

文章浏览阅读3.3k次。原文链接:http://www.xuanyusong.com/archives/24051.打包prefabusing UnityEngine;using System.Collections;using UnityEditor;//选择的物体放在不同的ABpublic class CreateAB_Respectively : MonoBehaviour {_解包出来的prefab怎么读取

推荐文章

热门文章

相关标签