OS复习笔记ch5-2

引言

在上一篇笔记中,我们介绍到了进程同步和进程互斥,以及用硬件层面上的三种方法分别实现进程互斥。其实,软件层面上也有四种方法,但是这些方法大部分都存在着一些问题:

  • “上锁”与“检查”是非原子操作(软件方法)
  • 都无法做到“让权等待”(软件+硬件方法)

接下来,我们介绍一种全新的信号量机制

信号量

  1. 原语
    信号量机制基于我们之前所说的原语操作,让用户通过使用操作系统提供的一对原语来对信号量进行操作,从而方便地实现进程互斥和进程同步。信号量(Semaphore)其实就是一个变量,它可以记录系统中某个资源的数量,而原语指的是 wait(S) 原语和 signal(S) 原语(或者说是 P 操作和 V 操作),可以看作是两个函数。

  2. 原理
    一个进程要获取临界资源时,先获取对应的信号量资源;
    当无信号量资源时,则该进程阻塞等待,进入等待队列。
    当有信号量资源时,则对该信号量资源进行P(-1)操作,然后获取该临界资源。
    当该进程使用完临界资源时,将释放信号量资源(对信号量资源进行V(+1)操作),然后唤醒等待队列中的进程。

  3. 分类1

  • 二元信号量
    实现互斥,采用二元信号量,即:该信号量的计数器,只能为0或1,又称为互斥量。
    这里的“元”指的是信号量的状态数目。

  • 多元信号量
    信号量说明: semaphore s=N;
    初始化指定一个非负整数值N,表示空闲资源总数(N>1时又称为“资源信号量”)

image.png

  • 强弱信号量
    强信号量: 最公平的策略是先进先出(FIFO)——被阻塞时间最久的进程最先从队列释放。 采用这个策略定义的信号量称为强信号量(strong semaphore)。
    弱信号量: 没有规定进程从队列中移出顺序的信号量称为弱信号量(weak semaphore)
  1. 分类2
  • 整型信号量
    信号量如果单纯是一个整数型的变量,那么就称为整型信号量,它的值记录了系统中某种资源的数量。在使用整型信号量的情况下,P 、V 操作是类似这样的:
int S = 1;
wait(int S)               
{                       while(S <= 0)			S = S-1              
}
signal(int S)
{S = S+1
}

image.png

很容易发现,由于PV操作原子性,整形信号量解决了引言的第一个问题,但整形信号量有一个很大的问题就是不能解决“让权等待”的问题,当资源不够时,进程会不停的执行While循环,无法及时让权。

  • 记录型信号量
    记录型信号量类似于数据库的一条记录,会有一些字段,最简单的就是下图中的semaphore结构体,有一个int类型的value,以及一个等待队列链表。
/*记录型信号量的定义*/
typedef struct {
//剩余资源数
int value;
Struct process *L;//等待队列
} semaphore;

同时,记录型信号量的 P、V 操作也有所不同,如下所示:

wait (semaphore S){S.value--if(S.value < 0){block(S.L) // 阻塞原语}
}
signal(semaphore S){S.value++if(S.value <= 0){wakeup(S.L) // 唤醒原语}
}
  • value 是可用的资源数,当它大于 0 的时候自然是存在可用资源(供大于求),当它小于 0 的时候,则说明不仅无可用资源而且有其他进程等着用(供不应求)。
  • 在进入区 value 一定会减一,表示申请到了资源,或者表示存在着某个进程有想要申请资源的意愿,无论是否能得到。
  • Block是一种原语,是无法获得资源的进程自己阻塞自己,将自己置于等待队列中,实现“让权等待”。
  • wakeup也是一种原语,是进程使用完资源后唤醒之前在该资源阻塞的其他进程
    image.png
    接下来,我们来看一个王道课上的例子理解一下记录型信号量的机制

image.png

假设现在有4个进程p0…p3需要使用打印机资源,但是只有两个打印机。
结构体中value的初始值是空闲资源数,这里就是2,等待队列初始为null

dd9c512f2db3971986d371d5d8cca1b.png

假设四个进程的执行顺序是p0→p1→p2→p3,那么p0先占据一个打印机资源,value–,同理p1占领另一个打印机,value–,此时的剩余资源数已经为0。

image.png

此时p3再申请打印机资源的时候,value–,value值小于0,执行阻塞原语,自己把自己阻塞,然后放到等待队列里,同理p4阻塞,在等待队列里面等待,value变为-2。

image.png

经过一段时间的使用,p0进程使用完打印机后释放资源signal(s),此时在等待队列的第一个进程p2获得打印机资源,由阻塞恢复就绪态,然后被OS选中进入运行状态,此时value为-1,等待队列中只有p3。

image-20240507120524798

假设p2进程执行较快,及时释放了打印机1,此时value为0,OS把打印机1分配给p3,p3变为就绪态,p2继续执行剩余代码然后退出。

image-20240507115414820
接着,CPU来到p1,p1使用完打印机2释放,value变为1,执行完剩余代码退出。

image-20240507120239713

然后,p3获得CPU,由就绪态变为运行态,执行相关代码,使用打印机。

image.png
p3释放打印机,执行结束后退出,记录型信号量恢复最初的状态

这里还有一个ppt上的例子,与之类似不再阐述,小伙伴们可以自行理解
image.png

总结

image.png

信号量机制是OS考察的重难点,无论是大题小题都会有所涉及,题目中的信号量一般默认情况指的都是记录型信号量,大家在学习的过程中要多动笔,多思考才能灵活运用。

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

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

相关文章

基于Springboot+Vue的Java项目-旅游网站系统开发实战(附演示视频+源码+LW)

大家好&#xff01;我是程序员一帆&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f49e;当前专栏&#xff1a;Java毕业设计 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f380; Python毕业设计 &am…

Meta最新研究: Flash Attention 为何是系统性能瓶颈?

I. 引言 随着机器学习趋向于更大和更复杂的模型,模型训练过程变得越来越计算和资源密集。生成式AI的出现进一步推动了模型开发的边界,大型语言模型(LLMs)通常在数百或数千个GPU上训练数月。以LLaMA2的70-B参数模型为例,需要1,720,320 GPU小时来训练。对于如此长的训练作业,训练…

高项第四版 十大管理及49个过程【背】作业分享

项目管理 1.十大管理【背】 包括&#xff08;口诀:范进整狗子&#xff08;沟质&#xff09; 才&#xff08;采&#xff09;干成疯子&#xff08;风资&#xff09;&#xff09;: &#xff08;1&#xff09;项目整合管理:识别、定义、组合、统一和协调各项目管理过程组的各个过…

了解外汇震荡类货币对特征与交易策略

外汇市场是全球最大的金融市场&#xff0c;每天的交易量超过6万亿美元。在这个市场上&#xff0c;货币对之间的价格变动反映了全球经济和政治动态。外汇货币对通常被分为三类&#xff1a;主要货币对、次要货币对和外来货币对。而在交易这些货币对时&#xff0c;市场表现通常分为…

自动化运维管理工具 Ansible-----【inventory 主机清单和playbook剧本】

目录 一、inventory 主机清单 1.1inventory 中的变量 1.1.1主机变量 1.1.2组变量 1.1.3组嵌套 二、Ansible 的脚本 ------ playbook&#xff08;剧本&#xff09; 2.1 playbook介绍 2.2playbook格式 2.3playbooks 的组成 2.4playbook编写 2.5运行playbook 2.5.1ans…

RabbitMQ的介绍和使用

1.同步通讯和异步通讯 举个例子&#xff0c;同步通讯就像是在打电话&#xff0c;因此它时效性较强&#xff0c;可以立即得到结果&#xff0c;但如果你正在和一个MM打电话&#xff0c;其他MM找你的话&#xff0c;你们之间是不能进行消息的传递和响应的 异步通讯就像是微信&#…

05-06 周一 Shell工程目录划分和开发最佳实践

05-06 周一 Shell工程目录划分和开发最佳实践 时间版本修改人描述2024年5月6日10:34:13V0.1宋全恒新建文档2024年5月6日11:07:12V1.0宋全恒完成 简介 之前楼主曾经完成过一个shell工程的开发&#xff0c;记得当时项目名称叫做campus-shell&#xff0c;主要是用来一键完成多个模…

经典面试题之滑动窗口专题

class Solution { public:int minSubArrayLen(int target, vector<int>& nums) {// 长度最小的子数组 // 大于等于 targetint min_len INT32_MAX;// 总和int sum 0;int start 0; // 起点for(int i 0; i< nums.size(); i) {sum nums[i];while(sum > targe…

TitanIDE安装常见问题解答

在软件开发和编程的世界里&#xff0c;集成开发环境&#xff08;IDE&#xff09;扮演着至关重要的角色。TitanIDE作为一款功能强大的开发工具&#xff0c;深受广大开发者的喜爱。然而&#xff0c;在安装和使用TitanIDE的过程中&#xff0c;开发者们往往会遇到一些问题和挑战。针…

毕业就业信息|基于Springboot+vue的毕业就业信息管理系统的设计与实现(源码+数据库+文档)

毕业就业信息管理系统 目录 基于Springboot&#xff0b;vue的毕业就业信息管理系统设计与实现 一、前言 二、系统设计 三、系统功能设计 1学生信息管理 2 公司信息管理 3公告类型管理 4公告信息管理 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设…

【阿道的成长日记】明确自己的人生主线,摆脱内耗低效能!

在职场中&#xff0c;许多人都会陷入这样的困境&#xff1a; &#x1f614;“对现在的工作厌倦不已&#xff0c;每天都感到痛苦&#xff0c;想要辞职。” &#x1f635;‍&#x1f4ab;“不确定自己适合从事哪种工作&#xff1f;现在这份工作是否真正适合我&#xff1f;” &…

C++ 数据内存分布揭秘:从栈到堆的探索之旅

目录 1. 栈(Stack) 2. 堆(Heap) malloc和new的区别 堆与栈在C中的异同点详解 3. 数据段(Data Segment) 4. 代码段(Code Segment) 5. 动态内存分配的陷阱 当我们谈论C编程时&#xff0c;对内存布局的理解至关重要。本文将深入探讨C中各种变量和数据结构在内存中的分布情况…