c++ 互斥锁

为练习c++ 线程同步,做了LeeCode 1114题. 按序打印:

给你一个类:

public class Foo {public void first() { print("first"); }public void second() { print("second"); }public void third() { print("third"); }
}

三个不同的线程 A、B、C 将会共用一个 Foo 实例。

  • 线程 A 将会调用 first() 方法
  • 线程 B 将会调用 second() 方法
  • 线程 C 将会调用 third() 方法

请设计修改程序,以确保 second() 方法在 first() 方法之后被执行,third() 方法在 second() 方法之后被执行。

提示:

  • 尽管输入中的数字似乎暗示了顺序,但是我们并不保证线程在操作系统中的调度顺序。
  • 你看到的输入格式主要是为了确保测试的全面性。

示例 1:

输入:nums = [1,2,3]
输出:"firstsecondthird"
解释:
有三个线程会被异步启动。输入 [1,2,3] 表示线程 A 将会调用 first() 方法,线程 B 将会调用 second() 方法,线程 C 将会调用 third() 方法。正确的输出是 "firstsecondthird"。

示例 2:

输入:nums = [1,3,2]
输出:"firstsecondthird"
解释:
输入 [1,3,2] 表示线程 A 将会调用 first() 方法,线程 B 将会调用 third() 方法,线程 C 将会调用 second() 方法。正确的输出是 "firstsecondthird"。

    提示:

    • nums 是 [1, 2, 3] 的一组排列

    答案&测试代码:

     

    #include <iostream>
    #include "listNode.h"
    #include "solution.h"
    #include <algorithm>
    #include <unordered_set>
    #include <unordered_map>
    #include <map>
    #include <string>
    #include <stdio.h>
    #include <stdlib.h>
    #include "solution3.h"
    #include "dataDefine.h"
    #include "uthash.h"
    #include "IntArrayList.h"
    #include <string.h>
    #include <thread>
    #include <atomic>
    #include "DemoClass.h"
    #include <mutex>
    #include <condition_variable>
    #include <functional>void printFirst() {std::cout << "first";
    }void printSecond() {std::cout << "second";
    }void printThird() {std::cout << "third";
    }void testLeeCode1114_() {class Foo { // 函数内部也可以定义类。private:std::mutex mtx; // 互斥锁std::condition_variable cv; // 条件变量bool isFirstDone;bool isSecondDone;public:Foo() {this->isFirstDone = false;this->isSecondDone = false;}void first(function<void()> printFirst) {{ // 加一对{}是为了限制下面加锁的作用域。离开作用域lock_guard自动立即释放锁// lock_guard构造时立即加锁(如果锁被占用会等待锁释放,一旦锁释放就抢占)。不支持手动释放锁。lock_guard 在析构时会自动释放锁。std::lock_guard<std::mutex> lock(mtx); // printFirst() outputs "first". Do not change or remove this line.printFirst();this->isFirstDone = true;}cv.notify_all(); // 唤醒所有等待锁的线程}void second(function<void()> printSecond) {// 立即加锁,同lock_guard, 但是unique_lock比较灵活,还支持延迟加锁。支持手动加锁、释放锁。需要手动管理。std::unique_lock<std::mutex> lock(mtx);// 判断是否满足执行条件。如果不满足就调用wait函数释放锁,该线程阻塞在这里等待被唤醒; 若满足执行条件则继续下面的代码逻辑。 // 被唤醒会判断是否满足执行条件,且满足条件则获取锁,然后继续下面的代码逻辑。不满足条件则继续等待。cv.wait(lock, [this](){return this->isFirstDone;}); // 这里的第二个参数是一个lambda表达式,表示一个匿名函数,该函数捕获this指针(函数体中用到该指针),没有参数。// printSecond() outputs "second". Do not change or remove this line.printSecond();this->isSecondDone = true;lock.unlock(); // 不要忘记释放锁。因为线程被唤醒后需要获取锁资源才会执行到这里,所以必须再释放,不能因为上面wait函数释放锁了就不调用释放了。cv.notify_all(); // 需要通知其他线程, 这里也可以调用notify_one, 只有一个线程在等待了。}void third(function<void()> printThird) {std::unique_lock<std::mutex> lock(mtx);cv.wait(lock, [this]() {return this->isSecondDone;});// printThird() outputs "third". Do not change or remove this line.printThird();lock.unlock();}};Foo foo; // 主线程生成Foo对象。// 因为线程要执行对象里的成员函数,所以第一个参数是函数指针, 第二个参数是该对象,后面的参数是该成员函数的传参,这里传递的函数std::thread t1(&Foo::first, &foo, printFirst);std::thread t3(&Foo::third, &foo, printThird);std::thread t2(&Foo::second, &foo, printSecond);t1.join();t2.join();t3.join();std::cout << endl << "finish" << endl;
    }

    执行结果:

    ok!

    提交到LeeCode:

    ok! 没问题。

    反面教材:

    void testLeeCode1114() { // LeeCode1114.按序打印. 反面教材,主线程加锁,子线程解锁,报错:unlock of unowned mutex 。 class Foo {mutex mtx_1, mtx_2;unique_lock<mutex> lock_1, lock_2;public:Foo() : lock_1(mtx_1, try_to_lock), lock_2(mtx_2, try_to_lock) {}void first(function<void()> printFirst) {printFirst();lock_1.unlock();}void second(function<void()> printSecond) {lock_guard<mutex> guard(mtx_1);printSecond();lock_2.unlock();}void third(function<void()> printThird) {lock_guard<mutex> guard(mtx_2);printThird();}};Foo foo; // 主线程生成Foo对象。std::thread t1(&Foo::first, &foo, printFirst); // 因为是线程要执行对象里的成员函数,所以第一个参数是函数指针, 第二个参数是该对象,后面的参数是该成员函数的传参,这里传递的函数std::thread t3(&Foo::third, &foo, printThird);std::thread t2(&Foo::second, &foo, printSecond);// 主线程等这3个线程执行结束:t1.join();t2.join();t3.join();// 报错: unlock of unowned mutex 
    }

     报错:unlock of unowned mutex 。 

    问题就在于主线程加锁, 然后子线程解锁。所以报错。线程必须先占有锁资源才能解锁。

    总结: 互斥锁就类比一个单人用的卫生间。一个人(线程)进去了会把卫生间锁住(加锁), 此时其他人(线程)想进去只能等待锁释放。

    如果一个人进去卫生间后发现不满足办事条件,比如没带纸(如示例代码中判断不满足执行条件),此时出去等待(如示例代码的wait函数释放锁),别人进去卫生间完事后出来通知说他用完了,然后刚才出去等待的那个人再次竞争到卫生间进去了, 然后会再次检查条件是否满足(如代码中的条件判断),发现卫生间有纸了,ok可以方便了。

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

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

    相关文章

    山东大学软件学院创新项目实训开发日志(20)之中医知识问答自动生成对话标题bug修改

    在原代码中存在一个bug&#xff1a;当前对话的标题不是现有对话的用户的第一段的前几个字&#xff0c;而是历史对话的第一段的前几个字。 这是生成标题的逻辑出了错误&#xff1a; 当改成size()-1即可

    WSL2-Ubuntu22.04下拉取Docker MongoDB镜像并启动

    若未安装docker可参考此教程&#xff1a;可以直接在wsl上安装docker吗&#xff0c;而不是安装docker desktop&#xff1f;-CSDN博客 1. 拉取镜像 docker pull mongo:latest 2.打开网络加速&#xff0c;再次拉取镜像 3.创建docker-compose.yml 进入vim编辑器后输入i进行编辑&a…

    中通 Redis 集群从 VM 迁移至 PVE:技术差异、PVE 优劣势及应用场景深度解析

    在数字化转型浪潮下&#xff0c;企业对服务器资源的高效利用与成本控制愈发重视。近期&#xff0c;中通快递将服务器上的 Redis 集群服务从 VM&#xff08;VMware 虚拟化技术&#xff09;迁移至 PVE&#xff08;Proxmox VE&#xff09;&#xff0c;这一技术举措引发了行业广泛关…

    Prometheus+Grafana实时监控系统各项指标

    一、监控架构设计 核心组件与数据流 Prometheus&#xff1a;时序数据采集、存储与告警规则管理Node Exporter&#xff1a;采集主机指标&#xff08;CPU、内存、磁盘、网络等&#xff09;数据库Exporter&#xff1a;如 mysqld_exporter、postgres_exporterGrafana&#xff1a;…

    [密码学基础]GMT 0029-2014签名验签服务器技术规范深度解析

    GMT 0029-2014签名验签服务器技术规范深度解析 引言 在数字化转型和网络安全需求激增的背景下&#xff0c;密码技术成为保障数据完整性与身份认证的核心手段。中国密码管理局发布的GMT 0029-2014《签名验签服务器技术规范》&#xff0c;为签名验签服务器的设计、开发与部署提…

    多路转接select服务器

    目录 select函数原型 select服务器 select的缺点 前面介绍过多路转接就是能同时等待多个文件描述符&#xff0c;这篇文章介绍一下多路转接方案中的select的使用 select函数原型 #include <sys/select.h> int select(int nfds, fd_set *readfds, fd_set *writefds, f…

    QT6 源(45):分隔条 QSplitter 允许程序的用户修改布局,程序员使用 IDE时,就是分隔条的用户,以及其 QSplitter 源代码

    &#xff08;1&#xff09; &#xff08;2&#xff09;本类的继承关系如下&#xff0c;所以说分隔条属于容器&#xff1a; &#xff08;3&#xff09;本类的属性&#xff1a; &#xff08;4&#xff09; 这是一份 QSplitter 的举例代码&#xff0c;注意其构造函数时候的传参&am…

    VSCode PIO使用Jlink SWD烧录Stm32

    一、背景 PIO的编译速度比Arduino快很多&#xff0c;同样支持Arduino的语法。VScode的自动补全和插件也能够帮助快速开发目前使用JLINK SWD的方式连接STM32 二、配置 在ini配置文件中&#xff0c;添加如下内容 [env:genericSTM32F103C8] platform ststm32 board genericS…

    JavaScript 渲染内容爬取:Puppeteer 入门

    在现代网络应用中&#xff0c;许多网页内容是通过 JavaScript 渲染生成的&#xff0c;传统的爬虫工具往往难以获取这些动态内容。Puppeteer 作为一种强大的浏览器自动化工具&#xff0c;为这一问题提供了优雅的解决方案。本文将带你入门 Puppeteer&#xff0c;介绍如何安装、启…

    卷积神经网络:视觉炼金术士的数学魔法

    引言&#xff1a;当数学遇见视觉炼金术 在人工智能的奇幻世界里&#xff0c;卷积神经网络&#xff08;CNN&#xff09;犹如掌握视觉奥秘的炼金术士&#xff0c;将原始像素的"铅块"淬炼成认知的"黄金"。这种融合数学严谨性与生物灵感的算法架构&#xff0c…

    Android Cordova 开发 - Cordova 快速入门(Cordova 环境配置、Cordova 第一个应用程序)

    一、Cordova 1、Cordova 概述 Cordova 是使用 HTML&#xff0c;CSS 和 JavaScript 构建混合移动应用程序的平台 2、Cordova 特征 &#xff08;1&#xff09;命令行界面&#xff08;Cordova CLI&#xff09; 这是可用于启动项目&#xff0c;构建不同平台的进程&#xff0c;…

    ubuntu18.04启动不了修复

    参考: 虚拟机里的Ubuntu18.4启动时进入到grub rescue救援模式&#xff08;无法正常进入到系统&#xff09;&#xff0c;ls查看后只有一个硬盘和分区&#xff0c;且无法找到/boot/grub文件【已解决】_ubuntu grub rescue-CSDN博客 本人fdisk错误使用,导致了grub启动不了 第一步…

    SpringBoot3设置maven package直接打包成二进制可执行文件

    注意事项 SpringBoot普通native打包顺序clean compile spring-boot:process-aot native:compile 使用以下配置只会的打包顺序clean package&#xff08;注意&#xff1a;使用此配置以后打包会有编译后的class文件、jar包、original源文件、二进制可执行文件【Linux是无后缀的包…

    【华为】防火墙双击热备-之-主备模式-单外网线路

    FW1和FW2的业务接口都工作在三层&#xff0c;上行连接二层交换机。上行交换机连接运营商的接入点&#xff0c;运营商为企业分配的IP地址为100.100.100.2。现在希望FW1和FW2以主备备份方式工作。正常情况下&#xff0c;流量通过FW1转发&#xff1b;当FW1出现故障时&#xff0c;流…

    MYSQL之表的操作

    1. 创建表 语法: CREATE TABLE table_name ( field1 datatype, field2 datatype, field3 datatype ) character set 字符集 collate 校验规则 engine 存储引擎; field 表示列名, datatype 表示列的类型character set 字符集, 如果没有指定字符集, 则以所在数据库的字符集为…

    RAG进阶:Chroma开源的AI原生向量数据库

    一、Chroma 核心概念与优势 1. 什么是 Chroma&#xff1f; Chroma 是一款开源的向量数据库&#xff0c;专为高效存储和检索高维向量数据设计。其核心能力在于语义相似性搜索&#xff0c;支持文本、图像等嵌入向量的快速匹配&#xff0c;广泛应用于大模型上下文增强&#xff0…

    店匠科技摘得 36 氪“2025 AI Partner 创新大奖”

    全场景 AI 方案驱动跨境电商数智化跃迁 4 月 18 日,36 氪 2025 AI Partner 大会于上海盛大开幕。大会紧扣“Super App 来了”主题,全力探寻 AI 时代的全新变量,探索 AI 领域下一个超级应用的无限可能性。在此次大会上,跨境电商独立站 SaaS 平台店匠科技(Shoplazza)凭借“店匠跨…

    SQL技术终极指南:从内核原理到超大规模应用

    一、DDL核心应用场景与最佳实践 1.1 表结构设计场景矩阵 业务场景核心语法要素典型实现案例电商用户画像JSON字段虚拟列索引CREATE TABLE users (id INT, profile JSON, AS (profile->>$.age) VIRTUAL, INDEX idx_age((profile->>$.age)))物联网时序数据分区表压…

    吴恩达深度学习作业CNN之ResNet实现(Pytorch)

    课程中认识许多CNN架构。首先是经典网络&#xff1a; LeNet-5AlexNetVGG 之后是近年来的一些网络&#xff1a; ResNetInceptionMobileNet 经典网络 LeNet-5 LeNet-5是用于手写数字识别&#xff08;识别0~9的阿拉伯数字&#xff09;的网络。它的结构如下&#xff1a; 网络…

    FPGA入门学习Day1——设计一个DDS信号发生器

    目录 一、DDS简介 &#xff08;一&#xff09;基本原理 &#xff08;二&#xff09;主要优势 &#xff08;三&#xff09;与传统技术的对比 二、FPGA存储器 &#xff08;一&#xff09;ROM波形存储器 &#xff08;二&#xff09;RAM随机存取存储器 &#xff08;三&…