百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

从零开始学Qt(79):基于互斥量的线程同步

liebian365 2024-10-17 14:01 29 浏览 0 评论

QMutex和QMutexLocker是基于互斥量的线程同步类,QMutex定义的实例是一个互斥量, QMutex主要提供3个函数。

  • lock():锁定互斥量,如果另外一个线程锁定了这个互斥量,它将阻塞执行直到其他线程解锁这个互斥量。
  • unlock():解锁一个互斥量,需要与lock()配对使用。
  • tryLock():试图锁定一个互斥量,如果成功锁定就返回true;如果其他线程己经锁定了这个互斥量,就返回false,但不阻塞程序执行。

基于QMutex的类定义

使用互斥量,对QDiceThread类重新定义(见从零开始学Qt(77):多线程程序实例 - 投骰子),不采用信号与槽机制,而是提供一个函数用于主线程读取数据。更改后的QDiceThread类定义如下:

class QDiceThread : public QThread
{
Q_OBJECT
public:
  QDiceThread();
  void diceBegin(); //掷一次骰子
  void dicePause(); //暂停
  void stopThread(); //结束线程
  bool readValue(int *seq, int *diceValue); //用于主线程读取数据的函数
protected:
	void run() override; //线程任务
private:
  QMutex mutex; //互斥量
  int m_seq=0; //掷骰子次数序号
  int m_diceValue; //骰子点数
  bool m_Paused=true; //暂停
  bool m_stop=false; //停止
};

QDiceThread类里用QMutex类定义了一个互斥量变量mutex。

定义了函数readValue(),用于外部线程读取掷骰子的次数和点数,传递参数采用指针变量, 以便一次读取两个数据。

基于QMutex的类实现

下面是QDiceThread类中关键的run()和readValue()函数的实现代码。

void QDiceThread::run()
{//线程任务
  m_stop=false; //启动线程时令 m_stop=false
  m_seq=0; //掷骰子次数
  qsrand(QTime::currentTime().msec());//随机数初始化,qsrand 是线程安全的
  while(!m_stop) //循环主体
  {
    if(!m_Paused)
    {
      mutex.lock();
      m_diceValue=qrand(); //获取随机数
      m_diceValue=(m_diceValue%6)+1;
      m_seq++;
      mutex.unlock();
    }
  msleep(500); //线程休眠 500ms
  }
  quit();//相当于exit(0),退出线程的事件循环
}

bool QDiceThread::readValue(int *seq, int *diceValue)
{
  if(mutex.tryLock()){
    *seq=m_seq;
    *diceValue=m_diceValue;
    mutex.unlock();
    return true;
  }
  else
  	return false;
}

在run()函数中,对重新计算骰子点数和掷骰子次数的3行代码用互斥量mutex的lock()和 unlock()进行了保护,这部分代码的执行就不会被其他线程中断。注意,lock()与unlock()必须配对使用。

在readValue()函数中,用互斥量mutex的tryLock()和unlock()进行了保护。如果tryLock()成功锁定互斥量,读取数值的两行代码执行时不会被中断,执行完后解锁;如果tryLock()锁定失败,函数就立即返回,而不会等待。

原理上,对于两个或多个线程可能会同时读或写的变量应该使用互斥量进行保护,例如 QDiceThread中的变量m_stop和m_paused,在run()函数中读取这两个变量,要在diceBegin()、diceEnd()和stopThread()函数里修改这些值,但是这3个函数都只有一条赋值语句,可以认为是原子操作,所以,可以不用锁定保护。

定义的互斥量mutex相当于一个标牌,可以这样来理解互斥量:列车上的卫生间一次只能进一个人,当一个人尝试进入卫生间就是lock(),如果有人占用,他就只能等待;等里面的人出来,腾出了卫生间是unlock(),这个等待的人才可以进入并且锁住卫生间的门,就是lock(),使用完卫生间之后他再出来时就是unlock()。

QMutex需要配对使用lock()和unlock()来实现代码段的保护,在一些逻辑复杂的代码段或可 能发生异常的代码中,配对就可能出错。

主线程调用子线程

这样定义的QDiceThread类,在主程序中只能调用其readValue()函数来不断读取数值。实例采用QMutex进行线程同步,其界面本系列上一篇文章中的完全相同,只是增加了定时器,用于定时主动去读取掷骰子线程的数值。 实例程序的Widget类的主要定义如下(省略了一些系统生成的声明):

class Widget : public QWidget
{
Q_OBJECT
public:
  Widget(QWidget *parent = nullptr);
  ~Widget();
protected:
	void closeEvent(QCloseEvent *event);
private:
  Ui::Widget *ui;
  QDiceThread threadA;
  QTimer mTimer;
  int mSeq,mDiceValue;
private slots:
  //自定义槽函数
  void on_threadA_started();
  void on_threadA_finished();
  void onTimeOut(); //定期器处理槽函数
};

主要是增加了一个定时器mTimer和其时间溢出响应槽函数onTimeOut(),在Widget的构造函数中将mTimer的timeout信号与此槽函数关联。

connect(&mTimer,SIGNAL(timeout()),this,SLOT(onTimeOut()));

onTimeOut()函数的主要功能是调用threadA的readValue()函数读取数值。定时器的定时周期设置为100ms,小于threadA产生一次新数据的周期(500ms),所以可能读出旧的数据,通过存储的掷骰子的次数与读取的掷骰子次数是否不同,判断是否为新数据。onTimeOut()函数的代码如下:

void Widget::onTimeOut()
{//定时器溢出处理槽函数
  int tmpSeq=0,tmpValue=0;
  bool valid=threadA.readValue(&tmpSeq, &tmpValue); //读取数值
  if(valid && (tmpSeq != mSeq) )//有效,并且是新数据
  {
    mSeq=tmpSeq;
    mDiceValue=tmpValue;
    QString str=QString::asprintf("第 %d 次掷散子,数为:%d", mSeq, mDiceValue);
    ui->plainTextEdit->appendPlainText(str);
    QPixmap pic; //图片显示
    QString filename=QString::asprintf(":/dice/pic/d%d.jpg", mDiceValue);
    pic.load(filename);
    ui->labPic->setPixmap(pic);
  }
}

窗口上几个按钮的代码如下(省略了按钮使能控制的代码):

void Widget::on_btnStartThread_clicked()
{//启动线程按钮
  mSeq=0;
  threadA.start();
}
void Widget::on_btnStopThread_clicked()
{//结束线程按钮
  threadA.stopThread(); //结束线程的 run ()函数执行
  threadA.wait();
}
void Widget::on_btnStartDice_clicked()
{//开始掷骰子按钮
  threadA.diceBegin();
  mTimer.start(100); //定时器 100ms 读取一次数据
}
void Widget::on_btnStopDice_clicked()
{//暂停掷骰子按钮
  threadA.dicePause();
  mTimer.stop(); //定时器暂停
}

基于QMutexLocker的类实现

QMutexLocker是另外一个简化了互斥量处理的类。QMutexLocker的构造函数接受一个互斥量作为参数并将其锁定,QMutexLocker的析构函数则将此互斥量解锁,所以在QMutexLocker实例变量的生存期内的代码段得到保护,自动进行互斥量的锁定和解锁。例如,QDiceThread的run() 函数的代码可以改写如下:

void QDiceThread::run()
{//线程任务
  m_stop=false; //启动线程时令 m_stop=false
  m_seq=0; //掷骰子次数
  qsrand(QTime::currentTime().msec());//随机数初始化,qsrand 是线程安全的
  while(!m_stop) //循环主体
  {
    if(!m_Paused)
    {
      QMutexLocker Locker(&mutex);
      m_diceValue=qrand(); //获取随机数
      m_diceValue=(m_diceValue%6)+1;
      m_seq++;
    }
    msleep(500); //线程休眠 500ms
  }
  quit();//相当于exit(0),退出线程的事件循环
}

其它与前面的相同,程序实现完全相同的功能。

————————————————

觉得有用的话请关注点赞,谢谢您的支持!

对于本系列文章相关示例完整代码有需要的朋友,可关注并在评论区留言!

相关推荐

深度解密epoll 如何工作的?(epoll基本处理流程)

epoll...

大乐透第19082期:头奖开出7注1000万分落六地 奖池41亿元

2019年7月17日晚开奖的体彩超级大乐透第19082期开奖号码为:前区06、18、20、21、31,后区03、04。本期大乐透前区号码五区比为1:0:3:0:1,二区和四区号码没有给出。当期前区和值...

【开奖】4月27日周六:福彩、体彩(2021年4月27日体彩开奖结果)

4月27日开奖福彩3D第2019110期:61222选5第2019110期:0812202122排列3第19110期:303排列5第19110期:30305大乐透第19047期:0304...

“红狒狒”落户哈尔滨铁路局(哈尔滨铁路红肠)

这几天,“红人”“红狒狒”在牡丹江机务段可引起了不小的轰动,众粉丝争相与其拍照留念,在该段人气爆棚!“红狒狒”到底何许人也?“红狒狒”,中文名:和谐3D型电力机车;绰号:红狒狒、番茄;制造商:大连机...

2D、3D、2.5D,做游戏还是搞噱头?玩家都晕了

前言游戏类型就像某种潮流,一种流行罢,另一种接棒成为主流。前两年的新作大多以“开放世界”为标签,在追求纯沙盒的过程中打造出一些细致的分类,比如说“类GTA沙盒”。诚然,纯碎的沙盒游戏并不多见,业内只有...

《战神4》PC版宣传片发布 GTX 1070即可60帧畅玩

在今年10月的时候索尼PlayStation官方正式宣布圣莫尼卡2018年的《战神4》将于2022年1月14日推出PC版本,官方在今天公布了一段PC版宣传片,并且公开了游戏的配置需求。下面让我们一起来...

男星深情好丈夫形象崩塌,半夜搂美女坐大腿,举止亲密

近日,于晓光被拍到深夜在酒吧玩,结束后与一名女子一起上车离开。上车后,女子直接坐在了他腿上,他也顺势搂着美女,美女满脸笑容地坐在他腿上玩手机离开。可能有人会好奇,于晓光是谁呢?于晓光是韩国艺人秋瓷炫的...

d3d12dll丢失怎么修复?d3d12dll加载失败怎么解决?

  d3d12.dll丢失怎么修复?d3d12.dll加载失败怎么解决?很多朋友想要运行游戏的时候都会遇到这个问题,这种情况该怎么办呢?今天系统之家小编给朋友们讲讲具体的解决方法,操作其实还蛮简单的。...

许多玩家反馈《生化4RE》PC一直崩溃 无法进入游戏

今日(3月24日),卡普空《生化危机4:重制版》正式发售,然而有部分PC玩家遇到了游戏崩溃等问题。很多玩家在贴吧发帖称游戏遇到了严重的崩溃问题,且经常反复,报错代码普遍为FatalD3Derror...

微软正式推出适用于WSL Linux的D3D12 GPU视频加速技术

今天,微软正式向WindowsSubsystemforLinux(WSL)用户发布了Direct3D12GPU视频加速支持。在微软通过WSL允许在Linux下使用Open...

《怪物猎人:崛起》曙光系统报错“Fatal d3d error”的解决办法

《怪物猎人:崛起》曙光系统报错“Fatald3derror”的解决办法不少小伙伴反应《怪物猎人:崛起》DLC曙光预载以后打不开游戏,出现了Fatald3derror类似的错误代码,这类问题的解...

Mac+双屏,前端程序员的专业配置 - Loctek 乐歌 D3D 双屏电脑显示器支架

做FE也有一段日子了,电脑屏幕每天在设计稿、浏览器、IDE、即时通讯工具、Terminal、邮箱之间切换。虽然mac的工作区带来了很多灵活,但是依然略显不足。于是入手支架,把公司配的电脑和显示器发挥起...

RPC 的原理和简单使用(rpc详解)

RPC的概念RPC,RemoteProcedureCall,翻译成中文就是远程过程调用,是一种进程间通信方式。它允许程序调用另一个地址空间(通常是共享网络的另一台机器上)的过程或函数。在调用的...

大厂开源的golang微服务rpc框架 — kitex

提前rpc估计所有的开发同学都知道,不知道的也无所谓,毕竟我也好几年没用了,今天带大家在复习一下。RPC(RemoteProcedureCall):远程过程调用,...

干货!一文掌握Protobuf所有语言所有用法,快收藏

说实话,Protobuf这个库,让人相见时难别亦难,东风无力百花残,每次等到要用它的时候,总感觉还没有完全掌握它的用法,而实际上等去百度或者谷歌的时候,教程都是多么的凌乱不堪。学会它,最直接关系到的,...

取消回复欢迎 发表评论: