互斥量、临界区、信号量、事件标志组和消息邮箱

2023-05-16

http://ejs90ejs.iteye.com/blog/1351642

互斥量、临界区、信号量、事件标志组和消息邮箱
2010年07月23日
          为了好的理解互斥量、临界区、信号量、事件标志组和消息邮箱,下面一些知识对初学者来说很重要:
  为了实现各任务之间的合作和无冲突的运行,在有关联的任务之间必须建立一些制约关系。这些制约关系主要有两种:直接制约关系和间接制约关系。
  直接制约关系源于任务之间的合作。例如,有两个任务:任务1和任务2,它们需要通过访问同一个数据缓冲区合作完成一项工作,任务1负责向缓冲区写入数据,任务2负责从缓冲区读取数据。显然任务1还未向缓冲区写入数据时(缓冲区为空),任务2因不能从缓冲区得到有效数据而应处于等待状态;只有等任务1向缓冲区写入了数据之后,才应该通知任务2去读取数据。相反,当缓冲区的数据还未被任务2读取时(缓冲区为满时),任务1就不能向缓冲区写入新的数据而应处于等待状态;只有等到任务2自缓冲区中读取数据后,才应该通知任务1去写入数据。显然,如果这两个任务不能如此协调工作,将势必造成严重的后果。
  间接制约关系源于对资源的共享。例如,任务1和任务2共享一台打印机,如果系统已经把打印机分配给了任务1,则任务2因不能获得打印机的使用权而应处于等待状态;只有当任务1把打印机释放后,系统才能唤醒任务2使其获得打印机的使用权。如果这两个任务不这样做,也会造成极大的混乱。
  由上可知,在多任务合作工作的过程中,操作系统应解决两个问题:一是各任务间应具有一种互斥关系,即对于某个共享资源,如果一个任务正在使用,则其他任务只能等待,等到该任务释放该资源后,等待的任务之一才能使用它;二是相关的任务在执行上要有先后次序,一个任务要等其伙伴发来通知,或建立了某个条件后才能继续执行,否则只能等到。
  任务之间的这种制约性的合作运行机制叫做任务间的同步(注意阿,是同步,不是同时啊!)
  
  1 互斥量(mutex)
  1.1 互斥量的定义
  互斥量是一种公共资源,在指定时刻,它只能被一个线程占有(也就是所有权特性),而且占有它的线程可以反复申请这个互斥量。
  1.2 使用互斥量的目的:适合用来控制线程对某些特定资源(比如,临界区和共享资源)
  ① 保护临界区(Critical Section):临界区是一段代码,在执行时不能被其他线程打断,如果要保护它,用互斥量可以实现。举例如下:
  status=tx_mutex_get(&my_mutex,TX_WAIT_FOREVER);//申请互斥量
  if(status!=TX_SUCCESS)  break;//判断是否申请成功
  tx_thread_slepp(5);//睡眠5个时钟节拍
  status=tx_mutex_put(&my_mutex);//释放互斥量
          以上代码就是临界区。
          ② 提供对共享资源的互斥访问:如果线程想对资源进行独占访问,就必须为每个资源提供对应的互斥量。而在访问资源之前必须为之申请到互斥量,用完之后要把互斥量释放掉,以便其他线程可以使用它来访问资源。
          1.3 使用互斥量需要注意的问题
          使用互斥量可能会带来死锁的问题和优先级倒置(反转)的问题。
          ⑴死锁的问题,比如:进程A要调用M资源,而M资源此刻却被进程B占用,且进程B要调用N资源而N资源此刻被A占用,两进程均向CPU发出资源申请,而不释放自己占用的资源,所以两进程均处于暂停状态,这时死锁问题。
          如何避免死锁问题呢?有其中一种方法就是:强制每个进程在同一时刻只能占有一个互斥量,如果一定要只有多个互斥量,可以要求线程必须按相同的顺序来申请互斥量,如:如果线程先得到互斥量1,然后才是互斥量2。
         如果线程进入了死锁状态,怎样从中恢复呢?有下面两种方法:
          ① 在调用tx_mutex_get()函数申请互斥量时,使用超时属性。
          ② 使用tx_thread_wait_abort()中止陷入死锁的线程的挂起状态。
         ⑵优先级倒置(反转)的问题,比如:一个低优先级的线程3某一时刻已经占有互斥量,而此时一个高优先级线程1申请此互斥量,此时级线程1就只能等待,这时如果有一个中优先级线程2已经就绪,那线程2就会中断线程3,从而就会出现线程2先于线程1而执行,这种现象就是优先级倒置问题。
         如何避免优先级倒置?可以让线程3申请的互斥量使用优先级继承的选项来解决这个问题。而优先级继承就是:当出现这种情况(一个低优先级的线程3某一时刻已经占有互斥量,而此时一个高优先级线程1申请此互斥量,此时级线程1就只能等待)时,低优先级线程3就继承高优先级线程的优先级,直到线程3释放互斥量后才恢复其优先级,这样就可以避免优先级倒置的问题。
          1.4 互斥量的一大特性:所有权特性。
  
  2 临界区(Critical Section)
         2.1临界区的定义
         通过对多线程的串行化来访问公共资源或一段代码,速度快,适合控制数据访问。
         2.2 使用临界区的目的
         同步同进程中的线程,保证在某一时刻只有一个线程能访问数据的简便方法
  ,在任意时刻只允许一个线程对共享资源进行访问。如果有多个线程试图同时访问临界区,那么在有一个进程进入后期他所有试图访问此临界区的线程将被挂起,并一直等到已经进入此临界区的线程离开。
         2.3 互斥量和临界区同步线程的区别
         虽然临界区同步速度很快,但却只能用来同步本进程内的线程,而不可用来同步多个进程中的线程。而互斥量比临界区复杂。因为使用互斥不仅仅能够在同一应用程序不同线程中实现资源的安全共享,而且可以在不同应用程序的线程之间实现对资源的安全共享。
  
  3 信号量(semaphore)
          3.1 信号量的定义
          每个信号量都是公共资源,其值是一个32位计数。信号量的数据结构为一个值和一个指针,指针指向等待该信号量的下一个进程。
  信号量的值与相应资源的使用情况有关。当它的值大于0时,表示当前可用资源的数量;当它的值小于0时,其绝对值表示等待使用该资源的进程个数。注意,信号量的值仅能由PV操作来改变。
          实现的P,V操作算法描述:
         P操作:while s>0 s=s-1
         V操作:s=s+1
         P表示申请一个资源,如果条件满足(即右可以分配的资源),则把资源分配给提出申请的进程,并且时资源数目s减1。V表示资源使用哪完毕之后,要把占有的资源释放,并且资源数目s加1 。
  
         3.2 使用信号的场合
         ⑴ 互斥访问,和互斥量类似,两者的区要区别是:信号量不支持所有权,而所有权是互斥量的核心。当信号量提供互斥访问时,信号量的值代表它允许多少个线程同时访问与之相联系的资源。大多数情况下,提供互斥量的信号量的初值为1,这意味着在任何时刻只有一个线程可以访问相关资源,其值只可能为1或0的信号量一般称为二进制信号量。
         使用二进制信号量时,使用者必须避免已经控制了信号量的线程再次调用tx_semaphore_get()来向信号量申请一个实例,如果调用了此项服务,这此调用失败,并将永久性挂起该线程,同时令资源永远不可用。
         ⑵ 事件通知:比如可以在生产者―消费者模式的程序中提供事件通知。在这种程序中,消费者在消费资源(例如队列中的数据)前试图获取信号量,生产者一旦生产了资源就增加信号量的值,也就可以理解为生产者把实例交给了信号量,而消费者把实例从信号量中拿走了。这类信号量通常初值为0,其值直到生产者生产了资源才会增加。
        上面这个例子说明了:生产者与消费者通过信号量通信,生产者通过向信号量放入实例产生事件通知(通过信号量的值变化来产生事件通知的作用),而消费者通过申请信号量实例来等待事件通知。
  ⑶ 线程同步:除了使用临界区与互斥量可以完成线程间的同步外,还可以使用信号量,其信号允许多个线程同时使用共享资源,它指出了同时访问共享资源的线程最大数目。在信号量内部有一个计数器,当有线程访问共享资源时,计数器将自动递减,当它为0时,不再允许其他线程对共享资的访问,直到有一个线程释放共享资源,从而完成对共享资源的保护。
         多个线程竞争n个资源(n>0),比如有4个资源,那么就初始化量为4,每当线程各自请求信号量的时候,如果信号量为0,则表示没有可用的资源,请求的线程要等它其他占用者释放信号量,如果信号量大于0,则线程的请求被允许,信号量被递减,然后线程可以按约定访问资源,当线程使用资源完毕,应该释放信号量,以便其他线程需要时使用
          3.3 互斥量和信号量的区别
         如果互斥对程序很关键,那么就该使用互斥量;如果互斥不是程序中的主要因素,那么就使用信号量,因为它比互斥量稍快,使用系统资源比较少。
          3.4 信号量的事件通知和线程同步的区别
          信号量的事件通知是用于两个线程之间(比如生产者―消费者问题),而信号量的线程同步时用于多线程之间(比如一个可以多人打电话的电话亭问题)。
  
   4 事件标志组
          4.1 事件标志组的理解
         事件标志可以被任何线程置位和复位,也可以被任何程序查看,线程可以因为等待一组事件标志被置位而挂起。每个事件标志由一个位代表,每32个事件标志被安排在一组。
          将事件标志置位或复位是通过把当前事件标志组和新的标志组进行逻辑“与”或“逻辑或”;为了得到事件标志,也要做类似的逻辑操作。一旦事件标志组中有标志被置位,系统就会检查相应组的挂起队列,如果能满足挂起线程,则该线程就被恢复。
          4.2 事件标志组的使用场合
          应为一个事件标志组中包括32个1位的标志,可以有非常多的组合方式,所有它比较适合于同步很多线程的场合。
  [/b]
  [b]5 消息队列
          5.1 如何理解消息队列
          首先它是一种队列,所以它是一种数据结构,但它又是“消息队列”,所以它是一种可以传递多个消息(消息也就是数据,实际上传递得是消息的指针)的数据结构。
          5.2 消息队列的使用
         如果线程与线程之间要传递很多消息(数据),那么为了适应不同数据的需要,最好是在存储器中建立多个数据缓冲区,把要传递的数据放在缓冲区中,其中在消息队列中分别存放消息的缓冲区的地址,这样就可实现线程间的数据通信。适合于线程间传递很多数据的场合
         5.3 消息队列的线程间通信和信号量的线程同步的区别
         消息队列的线程间通信是用于传递数据,而信号量的线程同步侧重的是信号量的指示作用。
  
  互斥量和二进制信号量的一个应用:
         一个一次只允许一个用户使用的电话亭,为了防止用户发生冲突,电话亭的门上就应该有这样一个标志,并用它来表示电话亭的被只有情况。例如,用一个可以变换两种颜色的牌子,用红色表示“有人”,用绿色表示“没人”。这样,人们见到牌子上的颜色是绿色时(线程通过查询知道信号量的值为1),就可以进去打电话(表示一个进程可以占用此资源(电话亭));如果是红色,那么就只好等待;如果又陆续来了很多人,那么就排队等待。其中,电话亭上的这个牌子就是信号量或互斥量。
  
  信号量同步线程的一个例子:
          一个电话亭可以允许多人(线程)打电话,电话亭门上的计数器在每进入一个人时自动减1,而每出去一个人时会自动加1。计数器上的初值就是电话亭的最多能容纳打电话的人数,那么来人只要见到计数器的值大于0,就可以进去打电话;否则只能等待。其中这个计数器就是用于同步线程的信号量。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

互斥量、临界区、信号量、事件标志组和消息邮箱 的相关文章

  • QT QCompleter 用法小结

    1 有何用处 QCompleter类提供基于项模型的补全功能 您可以使用QCompleter在任何Qt小部件 xff08 如QLineEdit和QComboBox xff09 中提供自动补全 当用户开始键入单词时 xff0c QComple
  • QT 关于全屏 与 退出全屏

    在某些场景 xff08 比如说 xff1a 车机系统 医疗仪器等 xff09 可能需要确保软件的全屏显示 xff0c 本次探讨的是如何实现全屏与退出全屏 实现全屏的几种方式 span class token operator lt span
  • QT 如何设置 QCompleter 样式

    在使用QCompleter xff0c 一般都会面临两个问题 xff1a 1 如何设置Completer的行高 xff1b 2 如何设置Completer的样式 本文将一次为你解答 xff0c 顺便讲讲ComboBox设置Completer
  • QT 如何知道用户点击 QTextBrowser 中的具体链接地址

    在通过鼠标点击 QTextBrowser 中的链接地址时 xff0c 如果是希望使用外部浏览器打开链接地址 xff0c 只需要添加一行代码 ui gt textBrowser gt setOpenExternalLinks true 但如果
  • QT 常用控件类型命名参考

    规范的定义 规范 xff0c 有名词 动词 代词等词性 意指符合逻辑 xff0c 客观 真实 全面 完整 准确 及时 xff0c 达标 明文规定或约定俗成的标准 如 xff1a 道德规范 技术规范等 或是指按照既定标准 规范的要求进行操作
  • C++ 如何让代码在main()函数之前或者之后执行

    程序最早执行的函数其实并不是main xff0c 在main函数之前会有一系列初始化的操作 xff08 通常是由链接器等完成的 xff09 比如在windows中 xff0c 这个函数是mainCRTStartup xff0c 这个函数是链
  • QT 如何提高 Qt Creator 的编译速度

    如何提高编译速度 xff0c 貌似是一个老生常谈的话题 对于Qter而言 xff0c 如何提高QT Creator 的编辑速度是一直都是大家所期盼的 本文也是查阅了各路大神的方法后整理出来的 xff0c 希望对各位有所帮助 1 在 pro文
  • 卧槽!出了一个Python实时目标跟踪系统神器!

    在当下自动驾驶 智慧城市 安防等领域对车辆 行人 飞行器等快速移动的物体进行实时跟踪及分析的需求可谓比比皆是 xff0c 但单纯的目标检测算法只能输出目标的定位 43 分类 xff0c 无法对移动的目标具体的运动行为及特征进行分析 xff0
  • C++ 什么是BCC校验

    BCC Block Check Character xff0c 俗称异或校验 BCC的实现方法 xff1a 将所有数据都和一个指定的初始值 通常是0 异或一次 xff0c 所得结果为校验值 BCC一般只是用来排错的 xff0c 并不是加密算
  • QT 如何让QLineEdit的文字从最左边开始显示

    当QLineEdit的文字长度超过了控件宽度 xff0c 一般而言 xff08 控件文字设为靠左对齐 xff09 xff0c 用户只能看到这一长串文字的后半段部分 但是如果想让QLineEdit从最左边开始显示字符串内容呢 xff1f 举个
  • QT setBuddy 用法小结

    将此标签的好友 xff08 窗口其他小部件 xff09 设置为好友 当用户按下此标签指示的快捷键时 xff0c 键盘焦点将转移到标签的好友小部件 好友机制仅适用于包含一个字符前缀为 amp 的文本的QLabels 此字符被设置为快捷键 设置
  • QT stackUnder 用法小结

    stackUnder是什么意思 xff1f 根据官方的解析 xff0c 简而言之一句话 xff1a 把窗口的小部件放置于父窗口的堆栈窗口中 本文将通过一则简单的示例 xff0c 来具体解析stackUnder该如何使用 官方解析 void
  • QT 线程安全的单例模式(使用Q_GLOBAL_STATIC实现)

    Qt提供了宏 Q GLOBAL STATIC xff08 官方说明文档 xff1a https doc qt io qt 5 qglobalstatic html xff09 xff0c 用于创建全局静态对象 鉴于此 xff0c 我们可以通
  • git push提示“fatal: the remote end hung up unexpectedly”的解决方法

    问题描述 在执行push命令时 xff0c 命令行提示 xff1a span class token function git span push origin master Enumerating objects 4968 done Co
  • 四旋翼无人机动力学模型及控制

    四旋翼无人机动力学模型及控制 I 欧拉角与旋转矩阵 Overview欧拉角与旋转矩阵Body Frame Angular Velocity and
  • Urbannav数据集/novatel_data/inspvax订阅

    使用Urbannav数据集做多原融合定位评估精度 xff0c 需要使用 novatel data inspvax作为真值 xff0c 但无法直接订阅该话题 rotopic echo novatel data inspvax 显示无法订阅 x
  • 软路由硬件, 研究了一圈还是J1900 , i211网卡又如何

    软路由的心路历程还真是纠结 xff0c 现写出来供大家参考 需求上也是几经翻车 xff1a 路由 xff0c NAS 之类的一体机是最先迸发出来也是最先被否的 本着大道至简 xff0c 设备专用的原则 软路由即路由 xff0c 不做他用 功
  • putty远程连接ubuntu18失败处理方法

    putty客户端远程连接报错 xff1a Network error Connection refused 解决方法 xff1a 1 先在Windows系统cmd命令下ping下此ip地址连接是否正常 xff0c 如果连接正常说明主机没有问

随机推荐

  • KITTI榜单新SOTA!相机-LiDAR双向融合新范式 | CVPR 2022 Oral & arXiv 2023

    点击下方卡片 xff0c 关注 CVer 公众号 AI CV重磅干货 xff0c 第一时间送达 点击进入 gt 计算机视觉 微信技术交流群 作者 xff1a 王利民 xff08 已授权转载 xff09 编辑 xff1a CVer https
  • C++ Primer 学习笔记 第七章 类

    233 定义类 span class token macro property span class token directive hash span span class token directive keyword include
  • 如何编写CMakeLists

    Preface 构建一个工程的时候 xff0c CMake需要知道的几个点 xff08 如果下面几件事你知道怎么做了 xff0c 多大的工程就都不是问题了 A 源代码在哪里 xff1f B 头文件在哪里 xff1f C 怎么生成静态或者动态
  • 【算法】最快最简单的排序——桶排序

    在我们生活的这个世界中到处都是被排序过的 站队的时候会按照身高排序 xff0c 考试的名次需要按照分数排序 xff0c 网上购物的时候会按照价格排序 xff0c 电子邮箱中的邮件按照时间排序 总之很多东西都需要排序 xff0c 可以说排序是
  • TCP、UDP数据包大小的限制

    1 概述 首先要看TCP IP协议 xff0c 涉及到四层 xff1a 链路层 xff0c 网络层 xff0c 传输层 xff0c 应用层 其中以太网 xff08 Ethernet xff09 的数据帧在链路层 IP包在网络层 TCP或UD
  • Jetson TX2刷机教程

    介绍 xff1a 本文介绍如何对Jetson TX2进行刷机 xff0c 系统版本为Jetpack4 6 0 准备 xff1a 主机 xff08 虚拟机 xff09 xff1a Ubuntu18 04Jetson TX2USB 连接线 刷机
  • 研扬Jetson NX镜像备份和恢复

    0 环境依赖 研扬RC S ARES 200AI NX CSC00型号Jetpack4 6 0 1 如何进入Recovery模式 1 先按住Recovery键再插电源上电 xff0c 此间一直按住Recovery键2 3秒之后松开 2 将U
  • SpringCloud-Eureka快速入门,集群搭建

    Eureka 个人主页 xff1a https blog csdn net hello list type 61 blog 前言 第一章 xff1a SpringCloud环境搭建 Rest使用 这里博主从说更新springcloud xf
  • SpringCloud-Ribbon和Feign快速上手

    Ribbon 个人中心 xff1a https blog csdn net hello list 前情提要 xff1a SpringCloud环境搭建 Rest使用SpringCloud Eureka快速入门 xff0c 集群搭建 首先我们
  • Java IO流详解

    Java IO流详解 个人主页 xff1a https blog csdn net hello list 今天我们来学习下java中的io部分 首先我们要知道io指的是什么 xff0c 输入输出 xff0c 就是输入输出流 xff0c 我们
  • SpringCloud-Hystrix服务熔断,快速入门

    Hystrix服务熔断 个人中心 xff1a 学习日记的博客 前情提要 xff1a SpringCloud环境搭建 Rest使用SpringCloud Eureka快速入门 xff0c 集群搭建SpringCloud Ribbon和Feig
  • 最新!!Intel首发UP SQUARED* GROVE物联网开发套件

    xfeff Intel官方网站正式推出UP SQUARED GROVE 物联网开发套件 https software intel com zh cn iot hardware up squared grove dev kit UP Squa
  • 再探指针:指针有什么用?(课堂笔记,来自翁恺老师的C语言进阶课,有着个人理解)

    指针的用处 xff1a 三个比较常用的场景 我们希望函数可以返回多个值的时候 我们都知道函数只可以return一个值 xff0c 使用指针便可以在一个函数里面返回多个值 举例 xff1a swap函数和数组中寻找最大值最小值的函数 xff0
  • java反射与注解详解,共同实现动态代理模式

    java反射与注解详解 xff0c 共同实现动态代理模式 个人主页 xff1a https blog csdn net hello list id xff1a 学习日记 不知不觉一年过去了 xff0c 整整一年 xff0c 这一年写了60多
  • 微信小程序快速入门

    微信小程序快速入门 在这里首先祝大家国庆节快乐 xff0c 其实原本文章都没有准备好 xff0c 也没有打算更文的 xff0c 那还是将就一下吧 xff0c 发个简单的 前言 相信大家对微信小程序并不陌生 xff0c 以前我们接触网络刚开始
  • RabbitMQ快速入门,这一篇看完教你学会

    RabbitMQ快速入门 今天学习RabbitMQ xff0c 你知道RabbitMQ是什么吗 xff0c RabbitMQ是一种消息中间件 xff0c 我们在写很多业务的时候 xff0c 有时候我们需要考虑到消息的实时性 xff0c 时效
  • 什么是协议栈

    协议栈是什么 1 协议栈是什么 简介 协议栈 xff0c 英语名称为Protocol stack xff0c 又称协议堆叠 xff0c 是计算机网络协议套件的一个具体的软件实现 协议套件中的一个协议通常是只为一个目的而设计的 xff0c 这
  • 任务,任务的切换,(TCB)

    任务也可以称作为进程 xff0c 是一个简单的程序 xff0c 该程序认为 CPU完全属于自己 xff0c 实时的应用的程序的设计的时候分割成了许多的任务 xff0c 每一个任务都对应应用的某一部分 每一个任务都被赋予一定的优先级 xff0
  • 浅谈pthread_setschedparam的使用

    浅谈pthread setschedparam 的使用 int pthread setschedparam pthread t target thread int policy const struct sched param param
  • 互斥量、临界区、信号量、事件标志组和消息邮箱

    http ejs90ejs iteye com blog 1351642 互斥量 临界区 信号量 事件标志组和消息邮箱 2010年07月23日 为了好的理解互斥量 临界区 信号量 事件标志组和消息邮箱 xff0c 下面一些知识对初学者来说很