线程同步-条件变量

news2024/9/21 20:40:34

在这里插入图片描述

文章目录

  • 引言
  • 条件变量
    • 初始化条件变量:pthread_cond_init
    • 销毁条件变量:pthread_cond_destroy
    • 条件等待:pthread_cond_wait
    • 唤醒等待:pthread_cond_signal、pthread_cond_broadcast
    • 认识条件变量
    • 接口使用

引言

有一个非常好的VIP自习室,一次只允许一个人进来,每一个自习完成的同学归还钥匙后,不能立马申请,第二次申请必须排队,也就是说其他人也必须排队,这种去自习室自习有一定的顺序性,这称之为同步。

换言之,每一个线程在访问临界资源时,有一定的顺序性,这称之为线程的同步。这里的顺序性可以是严格的顺序性,也可以是宏观上的具有相对顺序性。

条件变量

一个条件变量是一个pthread_cond_t类型

初始化条件变量:pthread_cond_init

int pthread_cond_init(pthread_cond_t *restrict cond,const pthread_condattr_t *restrict
attr);

cond:要初始化的条件变量
attrNULL

销毁条件变量:pthread_cond_destroy

int pthread_cond_destroy(pthread_cond_t *cond)

在调用 pthread_cond_destroy 之前,确保没有线程在等待这个条件变量

条件等待:pthread_cond_wait

int pthread_cond_wait(pthread_cond_t *restrict cond,pthread_mutex_t *restrict mutex);

cond:要在这个条件变量上等待
mutex:互斥量

唤醒等待:pthread_cond_signal、pthread_cond_broadcast

唤醒一个线程:

int pthread_cond_signal(pthread_cond_t *cond);

唤醒所有线程:

int pthread_cond_broadcast(pthread_cond_t *cond);

认识条件变量

有两个人分别是A和B,B往箱子中放苹果,A从箱子中拿苹果,A必须在B放完后拿苹果。箱子实际上是一个共享资源,苹果相当于数据,当B没有往共享资源中写数据时,A进行读取,那么读取的内容就是垃圾数据。A,B在进行操作的时候,必须保证箱子(共享资源)的安全,因此需要加锁pthread_mutex_lock ,完成之后进行解锁pthread_mutex_unlock。注定A,B需要互斥式的访问箱子(共享资源),这样才能保证B正在写的时候,A不会读取。A需要先检测箱子(共享资源)重是否有苹果(数据),当检测到没有苹果(数据),A需要解锁退出,然后A立马重新申请锁,再去检测,因此A可能在不断申请锁、解开锁。A如此频繁的申请锁,B可能抢不到锁。因此这种做法不合理:A、B互斥,如果B竞争能力弱的话,B抢不到锁,A只能频繁申请锁、解开锁。

在这里插入图片描述

为了解决这个矛盾,在上述场景中,引入一个铃铛,当A检测到箱子(共享资源)中没有苹果(数据),A解开锁之后不要再去申请锁,而是去铃铛那里等待铃铛,B放完苹果后摇铃铛,听到铃铛响了,A再去申请锁。

在这里插入图片描述

这里引入的铃铛就是条件变量,条件变量必须提供两个东西:

  1. 需要一个线程队列
  2. 需要有通知机制

此时又来一个C,也是来拿苹果,A和C就会形成竞争了,铃铛想起的时候,就会把A和C都唤醒,这就是pthread_cond_broadcast
在这里插入图片描述

接口使用

#include<iostream>
#include<string>
#include<unistd.h>
#include<pthread.h>

const int num=5;
pthread_mutex_t gmutex=PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t gcond=PTHREAD_COND_INITIALIZER;

void *Wait(void *args)
{
    std::string name=static_cast<const char*>(args);
    while (true)
    {
        pthread_mutex_lock(&gmutex);
        //条件等待
        pthread_cond_wait(&gcond,&gmutex);
        usleep(10000);
        std::cout<<"I am : "<<name<<std::endl;
        
        pthread_mutex_unlock(&gmutex);
        //usleep(100000);
    }
    
}

int main()
{
    pthread_t threads[num];
    for(int i=0;i<num;i++)
    {
        char *name=new char[1024];
        snprintf(name,1024,"thread-%d",i+1);
        pthread_create(threads+i,nullptr,Wait,(void *)name);
    }
    //唤醒线程
    while(true)
    {
        pthread_cond_signal(&gcond);   //一次唤醒一个线程
        //pthread_cond_broadcast(&gcond);   //一次唤醒多个线程
        std::cout<<"唤醒一个线程..."<<std::endl;
        sleep(2);
    }

    for(int i=0;i<num;i++)
    {
        pthread_join(threads[i],nullptr);
    }

    return 0;
}

在这里插入图片描述

为什么pthread_cond_wait在加锁和解锁之间使用?

确保条件检查的原子性:在多线程环境中,条件变量通常与互斥锁一起使用来保护共享资源。线程在检查条件之前需要持有锁,以避免其他线程修改共享资源。调用 pthread_cond_wait 时,函数会释放锁以让其他线程可以修改共享资源,然后在条件满足后重新获取锁,这样可以保证在条件变量被触发后,线程能够再次安全地检查条件和访问共享资源。
避免竞争条件:如果 pthread_cond_wait 不释放锁,那么其他线程将无法获取这个锁并修改条件,这可能导致死锁或线程无法继续工作。通过在 pthread_cond_wait 内部释放和重新获取锁,确保了条件检查的完整性和线程的正确同步。

在这里插入图片描述

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

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

相关文章

树莓派3B运行rasa init和rasa shell遇到的tensorflow报错总结

终于在我的树莓派上安装rasa-1.4.0版本成功&#xff08;见《树莓派智能语音助手之聊天机器人-RASA》&#xff09;。不过&#xff0c;在初始化rasa的时候还是遇到了很多报错&#xff0c;在此总结&#xff0c;供朋友们参考。 1. ModuleNotFoundError: No module named ‘tensorf…

合肥网站制作服务

合肥网站制作服务&#xff1a;为您的企业搭建数字化桥梁 随着互联网的迅猛发展&#xff0c;越来越多的企业意识到拥有一个专业的网站对于提升品牌形象和扩大市场影响力的重要性。合肥作为安徽省的省会&#xff0c;经济发展迅速&#xff0c;许多企业渴望通过优质的网站制作服务&…

Windows采用VS2019实现Open3D的C++应用

1、参考链接 https://blog.csdn.net/qq_31254435/article/details/137799739 但是&#xff0c;我的方法和上述链接不大一样&#xff0c;我是采用VS2019进行编译的&#xff0c;方便在Windows平台上验证各种算法。 2、创建一个VS2019的C Console工程 #include <iostream>…

Unity 6 预览版正式发布

Unity 6 预览版发布啦&#xff0c;正式版本将于今年晚些时候正式发布&#xff01; 下载链接&#xff1a; https://unity.com/releases/editor/whats-new/6000.0.0 Unity 6 预览版是 Unity 6 开发周期的最后一个版本&#xff0c;在去年 11 月 Unite 大会上&#xff0c;我们宣…

应急响应-DDOS-典型案例

某单位遭受DDoS攻击事件如下 事件背景 2019年2月17日&#xff0c;某机构门户网站无法访问&#xff0c;网络运维人员称疑似遭受DDoS攻击&#xff0c;请求应急响应工程师协助。 事件处置 应急响应工程师在达到现场后&#xff0c;通过查看流量设备&#xff0c;发现攻击者使用僵…

日拱一卒(4)——leetcode学习记录:路经总和

一、任务&#xff1a; 给你二叉树的根节点 root 和一个表示目标和的整数 targetSum 。判断该树中是否存在 根节点到叶子节点 的路径&#xff0c;这条路径上所有节点值相加等于目标和 targetSum 。如果存在&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 叶…

RCE与SQL漏洞的复现

eval长度限制突破 php eval函数参数限制在16个字符的情况下&#xff0c;如何拿到webshell呢 首先&#xff0c;我们还是先把环境搭好&#xff08;此次的所有漏洞环境我都部署在Ubuntu下,我的Ubuntu下的环境是nginxphp&#xff0c;至于Ubuntu下的nginxphp环境我也有博客也写到&a…

配置maven环境(全局)

全局设置 修改Maven 更改JRE 更改Java Compiler 修改完成后&#xff0c;Apply即可

一些有趣的XSS注入GAME

目录 1.ma,Spaghet 2.Jefff 3.Ugandan Knuckles 4.Ricardo Milos 5. Ligma 6.Ah Thats Hawt 7.Msfia 8.Ok,Boomer 1.ma,Spaghet innerHTML :设置 innerHTML 的值可以让你轻松地将当前元素的内容替换为新的内容。 HTML 5 中指定不执行由 innerHTML 插入的 <script&g…

【win/mac】矢量图形编辑软件Adobe Illustrator(AI) 2024版本下载与安装

一、Adobe AI 软件简介 什么是 Adobe AI 软件Adobe Illustrator&#xff08;简称 AI&#xff09;是一款功能强大的矢量图形编辑软件&#xff0c;广泛应用于平面设计、插画绘制、图标设计、包装设计等领域。它允许用户创建、编辑和处理各种复杂的矢量图形&#xff0c;具有极高的…

Metasploit 入门教程(非常详细)从零基础入门到精通,看完这一篇就够了!

一、引言 Metasploit 是一款强大的开源渗透测试工具框架&#xff0c;被广泛应用于网络安全领域。无论是安全研究人员、渗透测试工程师&#xff0c;还是对网络安全感兴趣的初学者&#xff0c;掌握 Metasploit 都是提升技能的重要一步。本教程将带您从零基础开始&#xff0c;逐步…

Linux软件编程学习第十三天

网络&#xff1a;1.协议&#xff1a;通信双方约定的一套标准 2.国际网络通信协议标准&#xff1a; 1.OSI协议&#xff1a; 应用层 发送的数据内容 表示层 数据是否加密 会话层 是否建立会话连接 传输层 …

HarmonyOS NEXT星河版零基础入门(3)

1. 系统弹出框 import { promptAction } from kit.ArkUI;Component struct weChat {State text:string build(){Column(){Text(this.text )Search({value:$$this.text})Button(改值).onClick(()>{promptAction.showToast({message:this.text,duration:10000})})}} } messa…

什么是DevUI?

DevUI是面向企业中后台产品的开源前端解决方案&#xff0c;其设计价值观基于"高效、开放、可信、乐趣"四种自然与人文相结合的理念&#xff0c;旨在为设计师、前端开发者提供标准的设计体系&#xff0c;并满足各类落地场景&#xff0c;是一款企业级开箱即用的产品。 …

Pikachu-XSS漏洞之cookie值获取、钓鱼结果和键盘记录实战记录

目录 Pikachu-XSS漏洞之cookie值获取、钓鱼结果和键盘记录实战记录 一、XSS&#xff08;get型&#xff09;之cookie值获取&#xff1a; 二、xss&#xff08;post型&#xff09;之cookie值获取 三、Xss之钓鱼攻击 四、XSS获取键盘记 Pikachu-XSS漏洞之cookie值获取、钓鱼结果…

RabbitMQ消息队列:概念、单节点和集群示例

目录 消息队列 概念 主流的消息队列 消息队列名词 &#xff08;1&#xff09;Broker &#xff08;2&#xff09;Topic &#xff08;3&#xff09;Producer &#xff08;4&#xff09;Consumer &#xff08;5&#xff09;Queue &#xff08;6&#xff09;Message 消息…

【C++二分查找】1011. 在 D 天内送达包裹的能力

本文涉及的基础知识点 C二分查找 LeetCode1011. 在 D 天内送达包裹的能力 传送带上的包裹必须在 days 天内从一个港口运送到另一个港口。 传送带上的第 i 个包裹的重量为 weights[i]。每一天&#xff0c;我们都会按给出重量&#xff08;weights&#xff09;的顺序往传送带上…

MVC与三层架构分层

1. 应用分层 应用分层类似公司的组织架构。我们进行项目开发时&#xff0c;最开始功能简单&#xff0c;前后端放在一起开发&#xff0c;但随着项目功能的复杂&#xff0c;项目会被前端和后端不同的团队接手&#xff0c;甚至更细粒度的团队。甚至后端开发也会根据功能再进行细分…

把VTK窗口嵌入到Qt的窗口中显示(二)

把VTK窗口嵌入到Qt的窗口中显示(二) 一、控件拖动 之前的文章介绍了,如何通过拖动一个控件,然后提升这个控件为QVTKWidget,然后来显示点云信息。 把VTK窗口嵌入到Qt的窗口中显示(一) 其实我们还可以直接通过代码来实现。我们打开QVTKWidget的代码可以看到,这个类就…

函数序列的逐点收敛和一致收敛的理解

目录 1. 逐点收敛(Pointwise Convergence) 1.1 逐点收敛定义 1.2 对逐点收敛的理解 1.3 举例说明 2. 一致收敛(Uniform Convergence) 2.1 一致收敛定义 2.2 对一致收敛的理解 2.3 举例说明 1. 逐点收敛(Pointwise Convergence) 1.1 逐点收敛定义 我们令 为…