当前位置: 首页 > news >正文

C++11之线程

编译环境:Qt

join:阻塞当前线程,直到线程函数退出

detach:将线程对象与线程函数分离,线程不依赖线程对象管理

注:join和detach两者必选其一,否则线程对象的回收会影响线程的回收,导致程序崩溃

思考:执行detach之后的线程如何退出?

  • 方法:设置标志位 
#include "dialog.h"
#include<iostream>
using namespace std;
#include <QDebug>
#include<thread>#include <QApplication>int add(int a,int b)
{qDebug()<<"add";qDebug()<<"a+b = "<<a+b;_sleep(1000);return a+b;
}class AA
{
public:AA(){//在构造函数中创建线程//如果使用_beginthreadex 线程函数是staticthread th(&threadFun,this);th.detach();}//执行detach之后线程如何退出//设置标志位void threadFun(){while(!m_isQuit){_sleep(100);qDebug()<<"do some work";}}
private:bool m_isQuit = false;
};int main(int argc, char *argv[])
{QApplication a(argc, argv);{thread th(&add,3,4);//线程函数可以以参数形式传入//join函数 阻塞当前线程 直到线程函数退出
//        _sleep(100);
//        qDebug()<<"before join";
//        th.join();
//        qDebug()<<"thread join";//detach 将线程对象与线程函数分离,线程不依赖线程对象管理qDebug()<<"before detach";th.detach();qDebug()<<"after detach";//注:join和detach两者必选其一,否则线程对象的回收会影响线程的回收,导致程序崩溃}Dialog w;w.show();return a.exec();
}

看下面一段代码:

执行结果不等于300,为什么?

  • ++g_value时分为三步:读,自增,写。在多线程情况下,可能存在多个线程读或写同一个值的情况,这就使得结果小于我们预期的值了。

线程并发引发的数据问题:

并发:同一时间间隔内,程序交替执行

解决:线程同步

 线程同步:同一时间,只允许一个线程访问资源

实现线程同步方法:

  • 互斥锁
  • 读写锁
  • 条件变量
  • 原子操作
  • 信号和槽
  • 事件循环

1、互斥锁

lock_guardunique_lock 管理互斥锁,让互斥锁使用更方便、更安全(可以避免死锁,比如我们使用完锁忘记释放了)

lock_guardunique_lock 遵循RAII

  • RAII,资源获取即初始化,是C++很重要的思想。

思考:如果需要在中途释放锁,怎么办?

  1. 使用 unique_lock
  2. 使用 lock_guard + 花括号实现,即锁一部分

锁锁定代码长度称为粒度,锁定的代码长度越长,锁的粒度越大,影响并发的效率。

2、条件变量

#include<condition_variable>
std::condition_variable con_var;

con_var.notify_one();(通知一个)

con_var.notify_all();(通知所有)

con_var.wait();

wait() 函数:

  • 有两个参数:第一个参数:是一个已经上锁的互斥锁(unique_lock),第二个参数:是一个可调用对象,其中包含函数指针,仿函数,bind ,lambda表达式
  • 如果这个函数执行返回值是false ,就会通知无效,可以避免误通知
  • 作用:阻塞当前线程,直到收到通知 notify_one notify_all
  • 当调用wait时,释放互斥锁,阻塞当前线程,将线程放入条件变量等待的容器中
  • 当收到通知时,获取互斥锁,执行后续代码
  • wait()和通知的关系:在使用的时候,一定是wait之后才能收到通知,否则会失效

 运行结果:

先打印出三个"before wait",之后每点击一下按钮,打印一个"after wait"

举例:

现在我们改变一下:

运行结果:

我们第一次按下按钮,quit取非为真,此时打印"after wait",再次按下,此时quit取非为假,此时无反应,第三次按下,此时quit又为真,打印"after wait"......

3、原子操作

#include<atomic>
atomic<int> cnt(0);

  • cnt++; 
  • cnt--;
  • cnt.load();
#include "dialog.h"#include<iostream>
using namespace std;#include <QDebug>
#include<thread>
#include <QApplication>#include<atomic>
atomic<int> cnt(0);void AutomicFunc()
{for(int i=0;i<100;i++){_sleep(10);//cnt++; //加锁的方式进行++,是原子操作,线程安全cnt = cnt + 1; //非原子操作,线程不安全_sleep(10);}
}int main(int argc, char *argv[])
{QApplication a(argc, argv);{thread th[3];for(int i=0;i<3;i++){th[i] = thread(&AutomicFunc);}for(int i=0;i<3;i++){th[i].join();}}qDebug()<<"count = "<<cnt.load();//取值Dialog w;w.show();return a.exec();
}

如有问题,欢迎交流指正! 


http://www.mrgr.cn/news/43693.html

相关文章:

  • 移动硬盘无法读取?详解原因与数据恢复方案
  • 秋招内推2025-招联金融
  • 国内目前顶级的哲学教授颜廷利:全球公认十个最厉害的思想家
  • 基础算法(5)——位运算
  • 快速理解mQ(三)——RabbitMQ 各种交换机的区别与应用
  • yolov11 部署瑞芯微rk3588、RKNN部署工程难度小、模型推理速度快
  • 从零开始构建大型语言模型——实现注意力机制
  • 各省份农林牧渔业总产值统计数据(2000-2022年)
  • Cisco Catalyst 9000 交换产品系列 IOS XE 17.15.1 发布下载,新增功能概览
  • 隐喻的使用及误用
  • InnoDB 磁盘结构 - Binlog
  • MySQL 管理
  • 鸿蒙harmonyos next flutter混合开发之开发plugin(获取操作系统版本号)
  • 资产管理系统建设方案,资产盘点,rfid,出入库,消耗品管理,系统方案,系统源码(word原件)
  • C++——用类实现输入和输出时间。要求:(1)私有数据成员包括时、分、秒。(2)输入输出功能改为由成员函数实现。(3)在类体内定义成员函数。
  • 父类的final方法能不能够被子类重写?
  • ChatGPT的150个角色提示场景实测(16)瑜伽师
  • AI学习指南深度学习篇-学习率衰减在深度学习中的应用
  • 华为OD机试 - 几何平均值最大子数(Python/JS/C/C++ 2024 E卷 200分)
  • vue实现token的无感刷新