江小南
2023/04/19阅读:44主题:萌绿
【操作系统】互斥锁、信号量机制
1. 互斥锁
解决临界区最简单的工具就是互斥锁(mutex lock)。一个进程在进入临界区时应获得锁;在退出临界区时释放锁。函数acquire()获得锁,而函数release()释放锁。
每个互斥锁有一个布尔变量available,表示锁是否可用。如果锁是可用的,调用acquire()会成功,且锁不在可用。当一个进程试图获得不可用的锁时,会被阻塞,直到锁被释放。
acquire(){
while(!available)
; // 忙等待
available=false; // 获得锁
}
release(){
available=true; // 释放锁
}
acquire()或release()的执行必须是原子操作,因此互斥锁通常采用硬件机制来实现。
互斥锁的主要缺点是忙等待,当有一个进程在临界区中,任何其他进程在进入临界区时必须连续循环调用acquire()。当多个进程共享同一CPU时,就浪费了CPU周期。因此,互斥锁通常用于多处理器系统,一个线程可以在一个处理器上等待,不影响其他线程的执行。
需要连续循环忙等的互斥锁,都可称为自旋锁(spin lock),如TSL指令、swap指令、单标志法。
特性:
-
需忙等。进程时间片用完才下处理机,违反“让权等待”。 -
优点:等待期间不用切换进程上下文,多处理器系统中,若上锁的时间段,则等待代价很低。 -
常用于多处理器系统,一个核忙等,其他核照常工作,并快速释放临界区。 -
不太适用于单处理机系统,忙等的过程中不可能解锁。

2. 信号量机制

1. 回顾
进程互斥的四种软件实现方式( 单标志法、双标志先检查、双标志后检查、Peterson算法)。
进程互斥的三种硬件实现方式(中断屏蔽方法、TS/TSL指令、 Swap/XCHG指令)
-
在双标志先检查法中,进入区的“检查”、“上锁”操作无法一气呵成,从而导致了两个进程有可能同时进入临界区的问题; -
所有的解决方案都无法实现“让权等待”。
1965年,荷兰学者Dijkstra提出了一种卓有成效的实现进程互斥、同步的方法——信号量机制。
2. 信号量机制
用户进程可以通过使用操作系统提供的一对原语来对信号量进行操作,从而很方便的实现了进程互斥、进程同步。
信号量其实就是一个变量(可以是一个整数,也可以是更复杂的记录型变量),可以用一个信号量来表示系统中某种资源的数量,比如:系统中只有一台打印机,就可以设置一个初值为1的信号量。
原语是一种特殊的程序段,其执行只能一气呵成,不可被中断。原语是由关中断/开中断指令实现的。软件解决方案的主要问题是“进入区的各种操作无法一气呵成”,因此如果能把进入区、退出区的操作都用“原语”实现,使这些操作能“一气呵成”就能避免问题。
一对原语:wait(S)原语和signal(S)原语,可以把原语理解为我们自己写的函数,函数名分别为wait和signal,括号里的信号量S其实就是函数调用时传入的一个参数。
wait、signal原语常简称为P、V操作(荷兰语proberen和verhogen)。因此,通常把wait(S)、signal(S)两个操作分别写为P(S)、V(S)。
3. 整型信号量
用一个整数型的变量作为信号量,用来表示系统中某种资源的数量。
说明:与普通整数变量相比:对信号量的操作只有三种,即 初始化、P操作、V操作。
例如:某计算机系统中有一台打印机。
4. 记录型信号量
整型信号量的缺陷是存在“忙等”问题,因此又提出了“记录型信号量”,即用记录型数据结构表示的信号量。
分析:
// 记录型信号量的定义
typedef struct{
int value; // 剩余资源数
struct process *L; // 等待队列
}semaphore;
// 某进程需要使用资源时,通过wait原语申请
void wait(semaphore S){
S.value--;
if(S.value < 0){
block(S.L);
}
}
// 进程使用完资源后,通过signal原语释放
void signal(semaphore S){
S.value++;
if(S.value <= 0){
wakeup(S.L);
}
}
wait(S)、signal(S)也可以记为P(S)、V(S),这对原语可用于实现系统资源的“申请”和“释放”。
S.value的初值表示系统中某种资源的数目。
对信号量S的一次P操作意味着进程请求一个单位的该类资源,因此需要执行S.value--,表示资源数减1,当S.value<0时表示该类资源已分配完毕,因此进程应该调用block原语进行自我阻塞(当前运行的进程从运行态->阻塞态),主动放弃处理机,并插入该类资源的等待队列S.L中。可见,该机制遵循了“让权等待”原则,不会出现“忙等”现象。
对信号量S的一次V操作意味着进程释放一个单位的该类资源,因此需要执行S.value++,表示该类资源加1,若加1后仍是S.value<=0,表示依然有进程在等待该类资源,因此应调用wakeup原语唤醒等待队列中的第一个进程(被唤醒进程从阻塞态->就绪态)。
3. 信号量机制的应用
信号量的值=这种资源的剩余数量(信号量的值如果小于0,说明此时有进程在等待这种资源)
P(S)——申请一个资源S,如果资源不够就则色等待。
V(S)——释放一个资源S,如果有进程在等待该资源,则唤醒一个进程。
1. 信号量机制实现进程互斥
信号量机制实现进程互斥:
-
分析并发进程的关键活动,划定临界区(如:对临界资源打印机的访问就应放在临界区,假设有一台)。 -
设置互斥信号量mutex,初值为1。 -
在进入区P(mutex)——申请资源。 -
在退出区V(mutex)——释放资源。
说明:信号量mutex表示“进入临界区的名额”。
注意:对不同的临界资源需要设置不同的互斥信号量。
P、V操作必须成对出现。缺少P(mutex)就不能保证临界资源的互斥访问。缺少V(mutex)会导致资源永不被释放,等待进程永不被唤醒。
// 信号量机制实现互斥
semaphore mutex=1; // 初始化信号量 说明:如果没有特别说明,可以把信号量的声明简写成这种形式,而不用写结构体
P1(){
...
P(mutex); // 使用临界资源前需要加锁
临界区代码段...
V(mutex); // 使用临界资源后需要解锁
...
}
P2(){
...
P(mutex);
临界区代码段...
V(mutex);
...
}
2. 信号量机制实现进程同步
进程同步:要让各并发进程按要求有序地推进。
信号量机制实现进程同步:
-
分析什么地方需要实现“同步关系”,即必须保证“一前一后”执行的两个操作(或两句代码)。 -
设置同步信号量S,初始为0。 -
在“前操作”之后执行V(S)。 -
在“后操作”之前执行P(S)。
技巧口诀:前V后P。
问题:P1和P2并发执行,由于存在异步性,因此二者交替推进的次序是不确定的。若P2的“代码段4”要基于P1的“代码段1”和“代码段2”的运行结果才能执行,那么我们就必须保证“代码段4”一定是在“代码段2”之后才会执行。如何做? 答案:
// 信号量机制实现进程同步
semaphore S=0; // 初始化同步信号量,初始值为0
说明:信号量S代表“某种资源”,刚开始是没有这种资源的。P2需要使用这种资源,而又只能由P1产生这种资源。 若执行到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(S)操作中执行wakeup原语,唤醒P2进程。这样P2就可以继续执行代码4了。
从上可以看出,无论先执行哪个进程,都能保证代码4在代码2之后执行。实现了进程同步。
3. 信号量机制实现前驱关系
进程P1中有句代码S1,P2中有句代码S2,P3中有句代码S3...P6中有句代码S6。这些代码要求按如下前驱图所示的顺序来执行: 其实每一对前驱关系都是一个进程同步问题(需要保证一前一后的操作),因此,
-
要为每一对前驱关系各设置一个同步信号量。 -
在“前操作”之后对相应的同步信号量执行V操作。 -
在“后操作”之前对相应的同步信号量执行P操作。
比如P2进程,必须等P1执行了V(a)之后才能执行。P6进程必须等P4执行了V(e)、P5进程执行了V(f)、P3进程执行了V(g)之后才能执行。
4. 小结
对于P(S)、V(S)的操作,除非特别说明,否则默认S为记录型信号量。
作者介绍