PV操作,信号量

例如,P1和P2是分别将数据送入缓冲B和从缓冲B读出数据的两个进程,为了防止这两个进程并发时产生错误,狄克斯特拉设计了一种同步机制叫“PV操作”,P操作和V操作是执行时不被打断的两个操作系统原语。执行P操作P(S)时信号量S的值减1,若结果不为负则P(S)执行完毕,否则执行P操作的进程暂停以等待释放。执行V操作V(S)时,S的值加1,若结果不大于0则释放一个因执行P(S)而等待的进程。对P1和P2可定义两个信号量S1和S2,初值分别为1和0。进程P1在向缓冲B送入数据前执行P操作P(S1),在送入数据后执行V操作V(S2)。进程P2在从缓冲B读取数据前先执行P操作P(S2),在读出数据后执行V操作V(S1)。当P1往缓冲B送入一数据后信号量S1之值变为0,在该数据读出后S1之值才又变为1,因此在前一数未读出前后一数不会送入,从而保证了P1和P2之间的同步。我国读者常常不明白这一同步机制为什么叫PV操作,原来这是狄克斯特拉用荷兰文定义的,因为在荷兰文中,通过叫passeren,释放叫vrijgeven,PV操作因此得名。这是在计算机术语中不是用英语表达的极少数的例子之一。
 
 
信号量是最早出现的用来解决进程同步与互斥问题的机制,包括一个称为信号量的变量及对它进行的两个原语操作。信号量为一个整数,我们设这个信号量为:sem。很显然,我们规定在sem大于等于零的时候代表可供并发进程使用的资源实体数,sem小于零的时候,表示正在等待使用临界区的进程的个数。根据这个原则,在给信号量附初值的时候,我们显然就要设初值大于零。 


p操作和v操作是不可中断的程序段,称为原语。P,V原语中P是荷兰语的Passeren,相当于英文的pass, V是荷兰语的Verhoog,相当于英文中的incremnet。 


P原语操作的动作是: 


(1) sem减1; 


(2) 若sem减1后仍大于或等于零,则进程继续执行; 


(3) 若sem减1后小于零,则该进程被阻塞后进入与该信号相对应的队列中,然后转进程调度。 






V原语操作的动作是: 


(1) sem加1; 


(2) 若相加结果大于零,则进程继续执行; 


(3) 若相加结果小于或等于零,则从该信号的等待队列中唤醒一等待进程,然后再返回原进程继续执行或转进程调度。 






需要提醒大家一点就是P,V操作对于每一个进程来说,都只能进行一次。而且必须成对使用。且在P,V愿语执行期间不允许有中断的发生。 


对于具体的实现,方法非常多,可以用硬件实现,也可以用软件实现。我们采用如下的定义: 


procedure p(var s:samephore); 

s.value=s.value-1; 
if (s.value<0) asleep(s.queue); 

procedure v(var s:samephore); 

s.value=s.value+1; 
if (s.value<=0) wakeup(s.queue); 

其中用到两个标准过程: 
asleep(s.queue);执行此操作的进程控制块进入s.queue尾部,进程变成等待状态 
wakeup(s.queue);将s.queue头进程唤醒插入就绪队列 
对于这个过程,s.value初值为1时,用来实现进程的互斥。 


虽软说信号量机制毕加锁方法要好得多,但是也不是说它没有任何的缺陷。由此我们也可以清晰地看到,这种信号量机制必须有公共内存,不能用于分布式操作系统,这是它最大的弱点
 
 
[二]P,V原语的应用 


正如我们在文中最开始的时候提到的,P,V原语不但可以解决进程管理当中的互斥问题,而且我们还可以利用此方法解决进程同步与进程通信的问题。 


(1)用P V原语实现进程互斥 


把临界区置于P(sem) 和V(sem)之间。当一个进程想要进入临界区时,它必须先执行P原语操作以将信号量sem减1,在进程完成对临界区的操作后,它必须执行V原语操作以释放它所占用的临界区。从而就实现了进程的互斥: 


具体的过程我们可以简单的描述如下: 


PA: 


P(sem) 





V(sem) 


PB: 


P(sem) 





V(sem) 


(2) 用P V原语实现进程同步 


进程同步问题的解决同样可以采用这种操作来解决,我们假设两个进程需要同步进行,一个进程是计算进程,另一个进程是打印进程,那么这个时候两个进程的定义可以表示为: 


PC(表示计算进程) 


A: local buf 


repeat 


buf=buf 


until buf=空 


计算 


得到计算结果 


buf=计算结果 


goto A 


PP:(表示打印进程) 


B: local pri 


repeat 


pri=buf 


until pri!=空 


打印buf中的数据 


清除buf中的数据 


goto B 


相应用P,V原语的实现过程为: 


PA: deposit(data) 


Begin local x 


P(bufempty) 


按FIFO方式选择一个空缓冲区buf(x) 


buf(x)=data 


buf(x)置满标记 


V(buffull) 


end 


PB:remove(data) 


Begin local x 


P(buffull) 


按FIFO方式选择一个装满 


数据的缓冲区buf(x) 


data=buf(x) 


buf(x)置空标记 


V(bufempty) 


end 


(3)用P V原语实现进程通信 


我们以邮箱通信为例说明问题: 


邮箱通信满足的条件是: 


<1>;发送进程发送消息的时候,邮箱中至少要有一个空格能存放该消息。 


<2>;接收进程接收消息时,邮箱中至少要有一个消息存在。 


发送进程和接收进程我们可以进行如下的描述: 


Deposit(m)为发送进程,接收进程是remove(m). Fromnum为发送进程的私用信号量,信箱空格数n。mesnum为接收进程的私用信号量,初值为0. 


Deposit(m): 


Begin local x 


P(fromnum) 


选择空格x 


将消息m放入空格x中 


置格x的标志为满 


V(mesnum) 


end 






Remove(m) 


Begin local x 


P(mesnum) 


选择满格x 


把满格x中的消息取出放m中 


置格x标志为空 


V(fromnum) 


end 


以上仅从最基本的进程问题上论述P,V原语的应用。当然关于这一部分的应用是十分广泛的。比如操作系统文化史上非常经典的哲学家就餐问题,生产-消费问题,读者-写者问题,理发师问题等等。大家不妨尝试一下用信号量的方法进行实现
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值