【操作系统】信号量机制(整型信号量、记录型信号量),用信号量实现进程互斥、同步、前驱关系

简介: 【操作系统】信号量机制(整型信号量、记录型信号量),用信号量实现进程互斥、同步、前驱关系

一、信号量机制

用户进程可以通过使用操作系统提供的一对原语来对信号量进行操作,从而很方便的实现了进程互斥、进程同步。

信号量其实就是一个变量(可以是一个整数,也可以是更复杂的记录型变量),可以用一个信号量来表示系统中某种资源的数量,比如:系统中只有一台打印机,就可以设置一个初值为1的信号量。

原语是一种特殊的程序段,其执行只能一气呵成,不可被中断。原语是由关中断/开中断指令实现的。软件解决方案的主要问题是由“进入区的各种操作无法一气呵成”,因此如果能把进入区、退出区的操作都用“原语”实现,使这些操作能“一气呵成”就能避免问题。

一对原语:wait(S)原语和signal(S)原语,可以把原语理解为我们自己写的函数,函数名分别为wait和signal,.括号里的信号量S其实就是函数调用时传入的一个参数。

wait、signal原语常简称为P、V操作(来自荷兰语proberen和verhogen)。因此,做题的时候常把wait(S)、signal(s)两个操作分别写为P(S)、V(S)【这是狄克斯特拉用荷兰文定义的,因为在荷兰文中,通过叫passeren,释放叫vrijgeven,PV操作因此得名。这是在计算机术语中不是用英语表达的极少数的例子之一。】

1.1 整型信号量

用一个整数型的变量作为信号量,用来表示系统中某种资源的数量(与普通整数变量的区别:对信号量的操作只有三种,即初始化、P操作、V操作)

Eg:某计算机系统中有一台打印机.…

int S = 1;      //初始化整型信号量$,表示当前系统中可用的打印机资源数
void wait(int S) {  //wait原语,相当于“进入区”
  while(S <= 0);  //如果资源数不够,就一直循环等待
  S = S - 1;    //如果资源数够,就占用一个资源
}
void signal (int S) { //signal原语,相当于“退出区”
  S = S + 1;      //使用完资源后,在退出区释放资源
}
进程P0:
...
wait(S);  //进入区,申请资源
使用资源    //临界区,访问资源
sinal(S); //退出区,释放资源
...

进程P0在使用资源之前,必须要先进行一个wait原语,对信号量S进行操作,如果当前资源已不够用,整型信号量S就是小于0的,此时P0就需要一直循环等待,若是P0进程执行wait时S大于0,就说明资源够用,此时资源的数量就要减一,此时P0就可以访问资源,在P0访问资源的同时,若是发生了进程切换,其他进程则需要等待P0释放资源才能访问,P0访问资源结束后,会进行signal原语,将整型信号量的值进行加一,也就是释放资源。


“检查”和“上锁”一气呵成避免了并发、异步导致的问题

【存在的问题:不满足“让权等待”原则,会发生“忙等】

1.2 记录型信号量

整型信号量的缺陷是存在“忙等”问题,因此人们又提出了“记录型信号量”,即用记录型数据结构表示的信号量。

/*记录型信号量的定义*/
typedef struct {
  int value;      //剩余资源数
  struct process *L;  //等待队列
}semaphore;
/*某进程需要使用资源时,通过wait原语申请*/
void wait(semaphore S) {
  S.value--;
  if (S.value < 0) {
    block(S.L);
  }
}

如果剩余资源数不够使用block原语使进程从运行态进入阻塞态,并把挂到信号量S的等待队列(即阻塞队列)中。

/*进程使用完资源后,通过signal原语释放*/
void signal (semaphore S) {
  s.value++;
  if (S.value <= 0) {
    weakup(S.L);
  }
}

释放资源后,若还有别的进程在等待这种资源,则使用wakeup原语唤醒等待队列中的一个进程,该进程从阻塞态变为就绪态。


S.value的初值表示系统中某种资源的数目。

对信号量S的一次P操作意味着进程请求一个单位的该类资源,因此需要执行S.vaue-,表示资源数减1,当S.value<0时表示该类资源己分配完毕,因此进程应调用bock原语进行自我阻塞(当前运行的进程从运行态→阻塞态),主动放弃处理机,并插入该类资源的等待队列SL中。可见,该机制遵循了“让权等待”原则,不会出现“忙等”现象。

对信号量S的一次V操作意味着进程释放一个单位的该类资源,因此需要执行S.vaue++,表示资源数加1,若加1后仍是S.value<=0,表示依然有进程在等待该类资源,因此应调用wakeup原语唤醒等待队列中的第个进程(被唤醒进程从阻塞态→就绪态)

二、用信号量实现进程互斥、同步、前驱关系

2.1 实现进程互斥

/*记录型信号量的定义*/
typedef struct {
  int value;      //剩余资源数
  struct process *L;  //等待队列
}semaphore;
/*信号量机制实现互斥*/
semaphore mutex = 1;  //初始化信号量
P1() {
  ...
  P(mutex);     //使用临界资源前需要加锁
  临界区代码段
  V(mutex);     //使用临界资源后需要解锁
}
P2() {
  ...
  P(mutex);     //使用临界资源前需要加锁
  临界区代码段
  V(mutex);     //使用临界资源后需要解锁
}
  • 分析并发进程的关键活动,划定临界区(如:对临界资源打印机的访问就应放在临界区)
  • 设置互斥信号量mutex,初值为1(理解:信号量mutex表示进入临界区的名额)
  • 在进入区P(mutex)一一申请资源
  • 在退出区V(mutex)一一释放资源
  • 注意:对不同的临界资源需要设置不同的互斥信号量。
  • P、V操作必须成对出现。缺少P(mutex)就不能保证临界资源的互斥访问。缺少V(mutex)会导致资源永不被释放,等待进程永不被唤醒。

2.2 实现进程同步

进程同步:要让各并发进程按要求有序地推进。

比如,P1、P2并发执行,由于存在异步性,因此二者交替推进的次序是不确定的。

若P2的“代码4”要基于P1的“代码1”和“代码2”的运行结果才能执行,那么我们就必须保证“代码4”一定是在“代码2”之后才会执行。

这就是进程同步问题,让本来异步并发的进程互相配合,有序推进。

/*信号量机制实现同步*/
semaphore S = 0;  //初始化同步信号量,初始值为0
P1() {
  代码1;
  代码2;
  V(S);
  代码3;
}
P2() {
  P(S);
  代码4;
  代码5;
  代码6;
}

若先执行到V(S)操作,则S++后S=1。之后当执行到P(S)操作时,由于S=1,表示有可用资源,会执行S-,S的值变回0,P2进程不会执行block原语,而是继续往下执行代码4。


若先执行到P(S)操作,由于S=0,S-后S=-1,表示此时没有可用资源,因此P操作中会执行block原语,主动请求阻塞之后当执行完代码2,继而执行V(S)操作,S+,使S变回0,由于此时有进程在该信号量对应的阻塞队列中,因此会在V操作中执行wakeup原语,唤醒P2进程。这样P2就可以继续执行代码4了(理解:信号量S代表“某种资源”,刚开始是没有这种资源的。P2需要使用这种资源,而又只能由P1产生这种资源)

2.3 实现进程的前驱关系

进程P1中有句代码S1,P2中有句代码S2,P3中有句代码S3.P6中有句代码S6。这些代码要求按如下前驱图所示的顺序来执行:



其实每一对前驱关系都是一个进程同步问题(需要保证一前一后的操作)

因此:

  • 要为每一对前驱关系各设置一个同步信号量
  • 在“前操作”之后对相应的同步信号量执行 V 操作
  • 在“后操作”之前对相应的同步信号量执行 P 操作
P1() {
  ...
  S1;
  V(a);
  V(b);
  ...
}
P2() {
  ...
  P(a);
  S2;
  V(c);
  V(d);
  ...
}
P3() {
  ...
  P(b);
  S3;
  V(g);
  ...
}
P4() {
  ...
  P(c);
  S4;
  V(e);
  ...
}
P5() {
  ...
  P(d);
  S5;
  V(f);
  ...
}
P4() {
  ...
  P(e);
  P(f);
  P(g);
  S6;
  ...
}



相关文章
|
1天前
|
存储 缓存 算法
探索现代操作系统的虚拟内存管理机制
【6月更文挑战第21天】在数字时代的浪潮中,操作系统作为计算机系统的核心,其设计和管理策略直接影响着计算效率和用户体验。本文将深入探讨现代操作系统中的虚拟内存管理机制,包括其工作原理、实现方式及其对系统性能的影响。通过分析虚拟内存技术如何优化资源分配、提高多任务处理能力及对硬件资源的抽象管理,揭示其在现代操作系统中的重要性和应用价值。
|
2天前
|
存储 负载均衡 算法
深入理解操作系统的进程调度
【6月更文挑战第20天】本文将探讨操作系统中的进程调度,包括其定义、重要性以及常见的调度算法。我们将通过具体的例子和代码片段来深入理解进程调度的工作原理和实现方式。最后,我们将讨论进程调度在现代操作系统中的应用和挑战。
|
2天前
|
存储 调度 C++
【操作系统】进程与线程的区别及总结(非常非常重要,面试必考题,其它文章可以不看,但这篇文章最后的总结你必须要看,满满的全是干货......)
【操作系统】进程与线程的区别及总结(非常非常重要,面试必考题,其它文章可以不看,但这篇文章最后的总结你必须要看,满满的全是干货......)
22 1
|
3天前
|
存储 算法
深入理解操作系统的内存管理机制
【6月更文挑战第19天】本文旨在探讨现代操作系统中内存管理的关键技术和策略,分析其对系统性能及稳定性的影响。通过介绍分页、分段、虚拟内存等概念,揭示操作系统如何有效管理物理与虚拟内存资源,以及这些技术在多任务处理和内存保护方面的应用。文章将重点讨论内存泄漏、碎片整理和页面置换算法等高级主题,以期为读者提供对内存管理复杂性及其解决方案的深刻理解。
8 2
|
4天前
|
算法 安全 调度
深入理解操作系统的虚拟内存管理机制
【6月更文挑战第18天】在现代计算机系统中,虚拟内存是实现多任务处理和内存保护的关键技术。本文将深入探讨操作系统如何通过虚拟内存管理机制来优化物理资源的使用,提高系统效率,并确保进程间的隔离与安全。我们将从虚拟内存的基本概念出发,逐步解析分页、分段、内存交换以及页面替换算法等核心组件,揭示它们如何协同工作以支撑起整个系统的内存需求。
|
4天前
|
调度
操作系统之进程调度机制
操作系统之进程调度机制
9 1
|
4天前
|
存储 缓存 运维
深入理解操作系统:从进程管理到内存分配
在数字时代的心脏,操作系统扮演着至关重要的角色。本文将深入探讨操作系统的核心机制,包括进程管理、内存分配和文件系统,揭示它们如何协同工作以支持现代计算需求。通过技术深度解析和实际应用示例,我们将一窥操作系统的复杂性与优雅,理解其在软件开发和系统性能优化中的重要性。
|
5天前
|
负载均衡 算法 调度
深入理解操作系统之进程调度
本文旨在探究操作系统核心机制之一——进程调度。文章首先概述进程与线程的基本概念,随后详细解析进程调度的目标、常见算法及其优缺点,并探讨现代操作系统中进程调度的高级话题,如多核调度和实时系统的调度策略。通过实例分析,本篇文章将帮助读者深化对进程调度复杂性的理解,并指出未来可能的发展方向。
|
8天前
|
监控 算法 安全
深入理解操作系统的内存管理机制
在数字时代的心脏,内存管理扮演着至关重要的角色。它是操作系统中的一项核心功能,负责协调、监控和控制计算机系统中的内存资源分配与回收。本文将深入探讨内存管理的基本原理、关键算法以及它在现代操作系统中的实现方式,揭示如何有效地利用和管理内存资源以优化系统性能和稳定性。
|
9天前
|
消息中间件 分布式计算 物联网
深入理解操作系统之进程与线程管理
操作系统的核心职责之一是进程与线程管理,它关乎系统的效率和稳定性。本文将剖析进程与线程的基本概念、生命周期以及它们在现代操作系统中的实现机制。通过对比分析,我们将揭示进程与线程的区别、优势及其适用场景,并探讨它们对系统性能的具体影响。进一步,文章将讨论进程间通信(IPC)的几种方式,以及同步和异步处理在多任务环境中的重要性。最后,我们将展望未来操作系统在进程与线程管理方面可能的发展趋势。