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

C++多线程同步(上)

多线程同步

  • 引言
  • 总述
  • 详情
    • 互斥锁
      • 示例
      • 运行结果
      • 分析
    • 条件变量
      • 示例一
        • 实现
        • 分析
        • 优化
        • 运行结果
      • 示例二
        • 实现代码
        • 运行结果
      • 示例三
        • 实现代码
        • 运行结果
    • 读写锁
      • 示例
        • 实现代码
        • 注意
        • 分析
        • 运行结果
      • 附言
        • 实现
        • 运行结果
        • 运行结果
        • 个人心得

引言

项目中使用多线程,会遇到两种问题,一种是对共享资源的访问时需要考虑多线程竞争访问导致的不是预期的结果,另一种是多线程之间需要同步的问题。其实质归根结底就是多线程之间的同步。
本文主要在C++11的基础之上,结合示例讲述多线程同步的几种方法。
本文为上篇。

总述

C++中多线程同步的方式分为:
互斥锁,条件变量,读写锁,信号量,future和promise,原子操作,线程局部存储

详情

下面根据上面提到的七种线程同步的方式分别给出示例。

互斥锁

互斥锁用于解决多线程间对共享资源的竞争问题,具有排它性。一旦一个线程获取锁,并加锁,其他的线程只能阻塞等待该线程解锁之后再抢占锁,且每次只能有一个线程获得锁,没有获得锁的线程只能阻塞等待。

示例

下面是互斥锁的示例:

#include <iostream>
#include <thread>
#include <mutex>using namespace std;
mutex g_mutex;void fun(int n,const char& c) {g_mutex.lock();cout << "子线程的线程id:" <<this_thread::get_id()<<"开始执行该线程......"<< endl;for (int i = 0; i < n;++i) {cout << c;}cout << endl;cout << this_thread::get_id()<<"子线程结束" << endl;g_mutex.unlock();
}int main()
{thread t1(fun,5,'S');thread t2(fun,6,'*');t1.join();t2.join();cout << "主线程的id:" << this_thread::get_id() << endl;return 0;
}

运行结果

在这里插入图片描述

分析

上面的示例创建了两个子线程,执行相同的线程处理函数,这就涉及到多线程对共享资源的竞争问题,这里两个子线程都抢着调用线程处理函数fun。由于何时加锁,在哪里加锁,需要结合开发人员的实际需求而定。这个示例希望程序能够输出完整的一个子线程调用fun函数后的内容,所以在刚进入线程处理函数和离开线程处理函数的时候进行加锁和解锁。

若是希望只给fun函数中的循环打印部分加锁,可以这样修改(只修改线程处理函数fun加锁,解锁位置,其它不变):

void fun(int n,const char& c) {cout << "子线程的线程id:" <<this_thread::get_id()<<"开始执行该线程......"<< endl;g_mutex.lock();for (int i = 0; i < n;++i) {cout << c;}g_mutex.unlock();cout << endl;cout << this_thread::get_id()<<"子线程结束" << endl;
}

执行的结果:
在这里插入图片描述
可以看到上面的示例,变动了加锁和解锁的位置之后,很明显的出现了资源竞争,输出后结果出现了混乱。当然输出结果也会出现很多种,无法确定。像下面这样,是再次运行被修改加锁和解锁的位置之后的运行结果。
在这里插入图片描述
也可能是这样的运行结果:
在这里插入图片描述
对于加锁的部分,当前获取锁的子线程可以保证其连续执行,但是不加锁的部分就会出现资源竞争抢占,最终两个子线程的同一个线程处理函数fun中不加锁的内容会穿插着输出,达不到想要的效果。

条件变量

条件变量需要与互斥锁搭配使用来达到想要的效果。

示例一

实现

下面是实现代码:

#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>using namespace std;
mutex g_mutex;
condition_variable g_cond;
bool IsReady = false;
const int g_num = 10;void fun(int n) {unique_lock<mutex> lock(g_mutex);while (!IsReady) {cout << "线程被阻塞....." << endl;g_cond.wait(lock);}cout << "线程" <<this_thread::get_id()<<"执行完成!"<< endl;
}void wakeUp() {this_thread::sleep_for(chrono::milliseconds(2));//延迟2毫秒,为了让子线程出现阻塞等待的过程unique_lock<mutex> lock(g_mutex);IsReady = true;cout <<
http://www.lryc.cn/news/307920.html

相关文章:

  • 猜猜心里数字(个人学习笔记黑马学习)
  • 实用Pycharm插件
  • 数据结构试题练习
  • s-table和columns初始化不完整,造成table文件的filter报错
  • SLA 是什么?如何实现 SLA 管理
  • 火灾安全护航:火灾监测报警摄像机助力建筑安全
  • JavaScript 基础学习笔记(五):函数、作用域、匿名函数
  • Qt环境配置VTK
  • 腾讯云最新活动_腾讯云促销优惠_代金券-腾讯云官网入口
  • 如何创建自己的Spring Boot Starter并为其编写单元测试
  • 数据分析---常见处理逻辑
  • 2024-02-26(金融AI行业概览与大数据生态圈)
  • git忽略某些文件(夹)更改说明
  • python爬虫实战:获取电子邮件和联系人信息
  • post请求同时上传文件并传递其他参数的前后端写法
  • 【数仓】基本概念、知识普及、核心技术
  • ky10-server docker 离线安装包、离线安装
  • Linux的gdb调试
  • IO多路复用-select模型
  • 班级事务管理系统设计与实现
  • 金三银四面试必问:Redis真的是单线程吗?
  • notejs+nvm+angular+typescript.js环境 Hertzbeat 配置
  • docker安装单机版canal和使用
  • qt_xml文件
  • 【DAY05 软考中级备考笔记】线性表,栈和队列,串数组矩阵和广义表
  • AutoGen Studio助力打造私人GPTs
  • SpringBoot 自定义映射规则resultMap association一对一
  • 华东地区汽车相关夹具配套企业分布图,你了解多少?
  • SpringBoot - 后端数据返回前端各个数据类型全局格式化
  • 实验室记账项目(java+Mysql+jdbc)