C++ 多线程(互斥锁、条件变量)

2024-08-28 05:52

本文主要是介绍C++ 多线程(互斥锁、条件变量),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

互斥锁
最简单示例:

#include <iostream>
#include <thread>
#include <mutex>std::mutex mtx;
int counter = 0;void increment() {for (int i = 0; i < 10000; ++i) {// b1 是创建出来的对象.  lock_guard 类似智能指针一样,为了防止忘记释放锁// 锁的管理范围看作用域范围 {}std::lock_guard<std::mutex> b1(mtx);counter++;}
}int main() {std::thread t1(increment);std::thread t2(increment);t1.join();t2.join();std::cout << "Counter value: " << counter << std::endl;return 0;
}

条件变量+互斥锁
互斥锁:保证只要一个线程在修改变量
条件变量:线程1与线程2之间需要一个同步信号,来统一控制线程是否该运行或者等待

#include <iostream>
#include <string>
#include <thread>  // 多线程
#include <vector>
#include <mutex>  // 互斥锁
#include <condition_variable>  // 条件变量
#include <chrono>  // 处理时间库//using namespace std;std::mutex mtx1;              // 互斥锁对象   
std::condition_variable cv_1; // 条件变量对象bool state = false;          // 条件变量 变量// 不使用传递引用
//void read_camera(std::string rtsp, std::string window_name) {// 使用传递引用避免拷贝
void read_camera(const std::string& rtsp, const std::string& window_name) {// 互斥锁 使用std::unique_lock<std::mutex> lck(mtx1);while (!state) {cv_1.wait(lck);  // 条件不成立 进入阻塞状态 释放掉锁 等待别人唤醒}for (int i = 0; i < 9999; i++) {std::cout << i << std::endl;}std::cout << "后启动的线程id:" << std::this_thread::get_id() << std::endl;std::cout << "rtsp" << rtsp << std::endl;std::cout << "window_name    " << window_name << std::endl;std::this_thread::sleep_for(std::chrono::seconds(2));  // 休眠2秒}void read_camera_1() {std::cout << "线程id:" << std::this_thread::get_id() << std::endl;std::this_thread::sleep_for(std::chrono::seconds(2));  // 休眠2秒// 互斥锁 使用std::unique_lock<std::mutex> lck(mtx1);state = true;cv_1.notify_all();  // 通知所有线程
}int main()
{ 单个线程启动//std::string param1 = "./jian_qian.mp4";//std::string param2 = "1";//std::thread t1([param1, param2]() {//    read_camera(param1, param2);//    });// 测试互斥锁 休眠2秒std::thread t([ ]() {read_camera_1();});// 循环启动线程// 要保证容器内的数量一致std::vector <std::string> camera_path{ "./jian_qian.mp4", "./jian_qian.mp4" };std::vector <std::string> camera_name{ "1", "2" };// 这里我用循环启动了2个线程,实际执行时,先会执行休眠2秒的函数,然后这两个线程会抢占运行先后顺序,// 假如第一个抢到了,第二个要等第一个运行完毕后才能运行,因为代码中互斥锁+条件变量 是只有一个线程可以运行,其他需要等待。std::vector<std::thread> threads;for (int i = 0; i < camera_path.size(); i++) {// 不传递引用//threads.push_back(std::thread([camera_path_i = camera_path[i], camera_name_i = camera_name[i]]() {//    read_camera(camera_path_i, camera_name_i);//    }));// 使用 std::ref 传递引用,避免值拷贝threads.push_back(std::thread(read_camera, std::ref(camera_path[i]), std::ref(camera_name[i])));}t.join();// 阻塞等待所有线程完成for (auto& t : threads) {// 避免重复启动线程,防止线程意外,未正确释放资源if (t.joinable()) {t.join();}}return 0;
}

这篇关于C++ 多线程(互斥锁、条件变量)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1113909

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Python之变量命名规则详解

《Python之变量命名规则详解》Python变量命名需遵守语法规范(字母开头、不使用关键字),遵循三要(自解释、明确功能)和三不要(避免缩写、语法错误、滥用下划线)原则,确保代码易读易维护... 目录1. 硬性规则2. “三要” 原则2.1. 要体现变量的 “实际作用”,拒绝 “无意义命名”2.2. 要让

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

C++右移运算符的一个小坑及解决

《C++右移运算符的一个小坑及解决》文章指出右移运算符处理负数时左侧补1导致死循环,与除法行为不同,强调需注意补码机制以正确统计二进制1的个数... 目录我遇到了这么一个www.chinasem.cn函数由此可以看到也很好理解总结我遇到了这么一个函数template<typename T>unsigned

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

GO语言短变量声明的实现示例

《GO语言短变量声明的实现示例》在Go语言中,短变量声明是一种简洁的变量声明方式,使用:=运算符,可以自动推断变量类型,下面就来具体介绍一下如何使用,感兴趣的可以了解一下... 目录基本语法功能特点与var的区别适用场景注意事项基本语法variableName := value功能特点1、自动类型推