std::thread和std::mutex

news2024/10/6 0:30:01

std::thread

std::thread用来创建一个线程:

#include <thread>

void threadFun(int temp)
{
	int i = 0;
}

int main()
{
	std::thread t1(threadFun, 100);
	t1.join();
	//t1.detach();
	return 0;
}

创建了一个名为t1的线程,调用join方法阻塞等待线程退出,也可以调用detach使线程处于游离态,参考Linux多线程,可以观察到,必须调用这两个方法中的一个,不然会报错:
在这里插入图片描述
看看thread析构函数:

~thread() noexcept {
    if (joinable()) {
        _STD terminate();
    }
}
    
_NODISCARD bool joinable() const noexcept {
    return _Thr._Id != 0;
}
    
void join() {
    if (!joinable()) {
        _Throw_Cpp_error(_INVALID_ARGUMENT);
    }

    if (_Thr._Id == _Thrd_id()) {
        _Throw_Cpp_error(_RESOURCE_DEADLOCK_WOULD_OCCUR);
    }

    if (_Thrd_join(_Thr, nullptr) != _Thrd_success) {
        _Throw_Cpp_error(_NO_SUCH_PROCESS);
    }

    _Thr = {};
}

void detach() {
    if (!joinable()) {
        _Throw_Cpp_error(_INVALID_ARGUMENT);
    }

    _Check_C_return(_Thrd_detach(_Thr));
    _Thr = {};
}

当thread对象被创建时,会给线程分配一个线程id(不为0),join和detach函数每次执行时会将_Thr对象置空,也就是将线程id置为0,当thread对象析构时,调用joinable函数发现线程id不为0就会terminate终止程序。

std::mutex

有关mutext相关知识可以参考Linux线程同步

mutex

有lock,try_lock,unlock三个方法。

recursive_mutex

有lock,try_lock,unlock三个方法。
它是可递归使用的互斥量,mutext对同一线程多次加锁会导致死锁,而recursive_mutex可以被同一个线程多次加锁而不会导致死锁。

#include <iostream>
#include <thread>
#include <mutex>

std::recursive_mutex rmtx;  // 定义一个可递归使用的互斥量对象

void printMessage(const std::string& message, int count) {
    std::lock_guard<std::recursive_mutex> lock(rmtx);  // 创建一个锁定互斥量的对象
    if (count > 0) {
        std::cout << message << " (" << count << ")" << std::endl;
        printMessage(message, count - 1);  // 递归调用
    }
}

int main() {
    std::thread t1(printMessage, "Hello", 3);

    t1.join();

    return 0;
}

printMessage方法中递归调用会重复加锁,递归退出会重复解锁。
需要注意的是,尽管std::recursive_mutex允许同一个线程多次加锁,但是相应地也必须多次解锁,每次解锁都要和加锁操作配对使用。否则,如果解锁次数多于加锁次数,那么仍然会导致死锁的问题

lock_guard

#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx;  // 定义一个互斥量对象

void printMessage(const std::string& message) {
    std::lock_guard<std::mutex> lock(mtx);  // 创建一个锁定互斥量的对象
    std::cout << message << std::endl;
    // 在作用域结束时,std::lock_guard对象会自动调用析构函数进行解锁操作
}

int main() {
    std::thread t1(printMessage, "Hello");
    std::thread t2(printMessage, "World");

    t1.join();
    t2.join();

    return 0;
}

lock_guard使用了RAII思想(资源获取就是初始化),构造函数里加锁,析构函数里面解锁,使用起来比较方便。

unique_lock

相比于lock_guard,提供了更精细的控制,感兴趣的可以看看源码

condition_variable

#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx;  // 定义一个互斥量对象
std::condition_variable cv;  // 定义一个条件变量对象
bool isReady = false;  // 全局变量,用于表示条件是否满足

void waitForSignal() {
    std::unique_lock<std::mutex> lock(mtx);  // 创建一个锁定互斥量的对象

    // 阻塞等待条件变量isReady为true
    cv.wait(lock, [] { return isReady; });

    // 条件满足后继续执行
    std::cout << "Received the signal" << std::endl;
}

void sendSignal() {
    std::this_thread::sleep_for(std::chrono::seconds(2));  // 延迟2秒钟,模拟耗时操作

    {
        std::lock_guard<std::mutex> lock(mtx);  // 创建一个锁定互斥量的对象
        isReady = true;  // 设置条件为true
    }
    cv.notify_one();  // 唤醒一个等待线程
}

int main() {
    std::thread t1(waitForSignal);
    std::thread t2(sendSignal);

    t1.join();
    t2.join();

    return 0;
}

条件变量可以用来控制当一个线程满足某种条件是,唤醒(notify_one)等待(wait)在条件变量上的其他线程

timed_mutex

#include <iostream>
#include <thread>
#include <mutex>

std::timed_mutex mtx;  // 定义一个 timed_mutex 对象

void doWork() {
    std::cout << "Thread " << std::this_thread::get_id() << " trying to lock" << std::endl;
    if (mtx.try_lock_for(std::chrono::seconds(2))) {
        std::cout << "Thread " << std::this_thread::get_id() << " locked successfully" << std::endl;
        std::this_thread::sleep_for(std::chrono::seconds(3));  // 模拟持有锁并进行一些操作
        mtx.unlock();  // 解锁
        std::cout << "Thread " << std::this_thread::get_id() << " unlocked" << std::endl;
    }
    else {
        std::cout << "Thread " << std::this_thread::get_id() << " failed to lock" << std::endl;
    }
}

int main() {
    std::thread t1(doWork);
    std::thread t2(doWork);

    t1.join();
    t2.join();

    return 0;
}

在上述示例中,定义了一个 std::timed_mutex 对象 mtx。doWork() 函数尝试在 2 秒的超时时间内获取互斥量的锁,如果成功获取锁,则持有锁并进行一些操作,然后解锁。如果在超时时间内无法获得锁,则放弃锁的获取。

std::timed_mutex 的主要方法是 try_lock_for(),该方法用于尝试在指定的时间段内获取互斥量的锁。如果在超时时间内成功获取锁,则返回 true,否则返回 false。

std::timed_mutex 的使用场景通常是在需要获取锁时,希望在一定时间内等待并放弃获取锁的线程,以避免长时间阻塞。它提供了一种更灵活的方式来控制线程对共享资源的访问。

recursive_timed_mutex

#include <iostream>
#include <chrono>
#include <mutex>
#include <thread>

std::recursive_timed_mutex mtx;  // 定义一个可递归加锁的互斥量对象

void recursiveFunc(int count)
{
    std::unique_lock<std::recursive_timed_mutex> lock(mtx, std::defer_lock);  // 创建 unique_lock 对象,但不立即锁定互斥量
    
    if (lock.try_lock_for(std::chrono::milliseconds(100))) {  // 尝试锁定互斥量,等待最多100毫秒
        std::cout << "Thread " << count << " acquired the lock." << std::endl;
        
        // 递归调用,多次获取锁
        if (count > 0) {
            recursiveFunc(count - 1);
        }
        
        lock.unlock();  // 解锁互斥量
        std::cout << "Thread " << count << " released the lock." << std::endl;
    } else {
        std::cout << "Thread " << count << " failed to acquire the lock within the timeout." << std::endl;
    }
}

int main()
{
    std::thread t1(recursiveFunc, 3);
    
    t1.join();
    
    return 0;
}

在上述示例中,定义了一个 std::recursive_timed_mutex 对象 mtx。在 recursiveFunc() 函数内部,通过创建一个 std::unique_lock 对象 lock 并使用 std::defer_lock 参数延迟锁定互斥量。然后,使用 try_lock_for() 函数尝试在最多 100 毫秒的时间内锁定互斥量。如果成功获取锁,则打印一条获取锁成功的消息,并递归调用 recursiveFunc() 函数来实现多次加锁。在每次加锁完成后,需要手动解锁。

总之,std::recursive_timed_mutex 类型提供了可递归加锁的互斥量机制,并且支持超时功能。它可以在某些场景下更方便地处理线程间的竞争和互斥问题。需要注意的是,在使用递归互斥量时需要小心避免出现死锁情况。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/745308.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【leetcode】61. 旋转链表 (python)

题目链接&#xff1a;61.旋转链表 写法一&#xff1a;暴力 class Solution(object):def rotateRight(self, head, k):""":type head: ListNode:type k: int:rtype: ListNode"""# 暴力&#xff1a;每次移动一个&#xff0c;时间复杂度 O(N^2)&am…

详解汽车电磁悬架(上)

摘要&#xff1a; 从汽车发明之日起&#xff0c;人们对于汽车平稳行驶的追求步伐就不曾停止过。充气轮胎的发明让车辆能够更加平稳舒适的行驶&#xff0c;承载式车身以及加厚的底盘则进一步提高了汽车的舒适性。随着科技的发展与生活水平的提高&#xff0c;人们对汽车的舒适性…

【C】字符串函数和内存函数的介绍

库函数&#xff08;这些函数都在头文件string.h中&#xff09; 字符串函数求字符串长度strlen 长度不受限的字符串函数strcpystrcmpstrcat 长度受限的字符串函数strncpystrncmpstrncat 字符串查找strstrstrtok 错误信息报告strerror 字符操作字符分类函数字符转换函数 内存函数…

测试必备的15个docker命令,你都掌握了吗

1、Docker容器信息 ##查看docker容器版本docker version##查看docker容器信息docker info##查看docker容器帮助docker --help 2、镜像操作 提示&#xff1a;对于镜像的操作可使用镜像名、镜像长ID和短ID。 2.1、镜像查看 ##列出本地imagesdocker images##含中间映像层docke…

iview-admin使用小结

首先在使用一个框架之前一定要完整的看一下相关文档&#xff0c;因为框架中会封装常用的功能&#xff0c;也会更加符合大众要求。在ui设计图上&#xff0c;可能实现某个功能设计图中给出的交互并不是很好&#xff0c;而在框架中有更好的组件可以实现&#xff0c;但因为没有看文…

物联网技术、测试要点和测试标准

目录 物联网定义 物联网中最常用的技术 物联网测试概述 测试人员在物联网中面临的挑战 测试物联网系统时使用了各种工具 &#x1f381;更多干货 完整版文档下载方式&#xff1a; 物联网定义 物联网&#xff1a;利用嵌入式电子设备、微芯片等连接车辆、家电、医疗设备&am…

广州华锐互动 | 鸡养殖VR教育实训系统

鸡养殖VR教育实训系统由广州华锐互动开发&#xff0c;是一种基于虚拟现实技术的培训工具&#xff0c;旨在为鸡养殖行业的从业人员提供更加直观、生动的学习体验。该系统通过虚拟现实技术&#xff0c;将鸡养殖过程中的各种场景、操作和技能呈现在学员面前&#xff0c;让学员可以…

关于 Linux 的常用命令

目录 前言什么是 Linux获得 Linux 环境为什么使用命令控制关于 XShell 下的复制粘贴 Linux 常用命令ls (罗列文件)cd (切换目录)pwd (显示当前路径)touch (创建一个空文件)cat (读取文件内容)man (查看用户手册)echo (回显)vim (文件编辑器)mkdir (创建目录)rm (删除目录/文件)…

数仓常见问题以及解决方案yyds

文章目录 01 数据仓库现状业务系统侧【上游】数仓内部数据应用测【下游】 02 如何解决问题业务系统侧【上游】数仓内部 03 体系化架构设计一致性维度一致性事实数据应用侧【下游】 建设可视化的取数平台 大家好&#xff0c;我是脚丫先生 (o^^o) 看到几篇不错的文章&#xff0c…

App Inventor 2 从.aia项目文件中提取.aix拓展包

从.aia项目文件中提取.aix拓展包的步骤 如果看到一个项目中&#xff0c;用到了拓展&#xff0c;但拓展的帮助中没有原链接&#xff0c;这时就可以考虑从项目文件中进行提取&#xff1a; 导出项目.aia文件&#xff1a; 在电脑的“下载”目录查看.aia项目文件&#xff1a; 将.a…

写在2023年年中

写在2023年年中 收拾旧山河再出发 很久没有写规划了&#xff0c;博客也很少写&#xff0c;着实是太懒了。想想自己年纪也不小了&#xff0c;加上程序员的环境越来越不好&#xff0c;总得想想自己的出路不是。 目标 提升系统设计能力保持算法刷题能力(一般面试需要)知识输出能…

RabbitMQ工作模式代码示例及用SpringBoot整合RabbitMQ

一&#xff0c;RabbitMQ的工作模式 RabbitMQ 的工作模式是指 RabbitMQ 中不同的消息传递方式&#xff0c;包括简单模式、工作队列模式、发布订阅模式、路由模式和主题模式 。这些工作模式适用于不同的应用场景。详细的文档可参照RabbitMQ的官网&#xff1a;RabbitMQ: easy to u…

基于JavaSwing+MySQL的员工工资管理系统

点击以下链接获取源码&#xff1a; https://download.csdn.net/download/qq_64505944/88042468?spm1001.2014.3001.5503 JDK1.8 MySQL5.7

虚拟与现实的交融:VR与数字孪生的融合之道

VR即虚拟现实&#xff08;Virtual Reality&#xff09;和数字孪生&#xff08;Digital Twin&#xff09;是当今科技领域的两大热门概念&#xff0c;VR以其沉浸式的体验和逼真的虚拟环境&#xff0c;让用户身临其境&#xff0c;开启了全新的交互方式和感官体验&#xff1b;而数字…

副业做什么比较靠谱,不想只靠一个收入

科思创业汇 大家好&#xff0c;这里是科思创业汇&#xff0c;一个轻资产创业孵化平台。赚钱的方式有很多种&#xff0c;我希望在科思创业汇能够给你带来最快乐的那一种&#xff01; 如今做一些副业能不能有一点安全感&#xff0c;分享一些可靠的副业&#xff0c;坚持下去&…

Abandoning the Bayer-Filter to See in the Dark 论文阅读笔记

这是CVPR2022的一篇暗图增强的文章&#xff0c;TCL AI Lab与福州大学&#xff0c;韩国延世大学&#xff0c;安徽大学的合作论文网络以黑暗环境下拍摄的color raw为输入&#xff0c;用一个de-bayer-filter module恢复无拜尔滤波器的raw data&#xff08;文章认为拜尔滤波器使得光…

leetcode 987. 二叉树的垂序遍历(java)

二叉树的垂序遍历 leetcode 987. 二叉树的垂序遍历题目描述DFS 优先队列&#xff08;堆&#xff09;代码演示 二叉树专题 leetcode 987. 二叉树的垂序遍历 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/vertical-orde…

AI文本标注的概念,类型和方法

我们每天都在与不同的媒介&#xff08;例如文本、音频、图像和视频&#xff09;交互&#xff0c;我们的大脑对收集到的信息进行处理和加工&#xff0c;从而指导我们的行为。在我们日常接触到的信息中&#xff0c;文本是最常见的媒体类型之一&#xff0c;由我们交流使用的语言构…

优化SQL查询实现高效数据检索(一)

大家好&#xff0c;SQL&#xff08;结构化查询语言&#xff09;可以帮助大家从数据库中收集数据&#xff0c;它是专为此而设计的&#xff0c;换句话说&#xff0c;它使用行和列来处理数据&#xff0c;让使用者能够使用SQL查询来操作数据库中的数据。 SQL查询 SQL查询是一系列…

11-2.箭头函数与函数的this问题

目录 1 箭头函数 1.1 基本使用 1.2 在只有一个参数的时候&#xff0c;小括号可以省略 1.3 只有一行时&#xff0c;可以省略大括号 1.4 只有一行时的return问题 1.5 直接返回一个对象 1.6 箭头函数使用剩余参数 2 this问题 2.1 直接打印this 2.2 fucntion定…