信号量P,V操作

本文详细介绍了信号量和PV操作的基本概念及其在进程同步与互斥中的应用,并通过几个典型示例阐述了如何利用信号量和PV操作解决实际问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

信号量是最早出现的用来解决进程同步与互斥问题的机制(也可实现进程通信),包括一个称为信 号量的变量及对它进行的两个原语操作。信号量为一个整数,我们设这个信号量为:sem。很显然,我们规定在sem大于等于零的时候代表可供并发进程使用的 资源实体数,sem小于零的时候,表示正在等待使用临界区的进程的个数。根据这个原则,在给信号量附初值的时候,我们显然就要设初值大于零。

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

且在P,V愿语执行期间不允许有中断的发生。

对于具体的实现,方法非常多,可以用硬件实现,也可以用软件实现。这种信号量机制必须有公共内存,不能用于分布式操作系统,这是它最大的弱点。

首先应弄清PV操作的含义:PV操作由P操作原语和V操作原语组成(原语是不可中断的过程),对信号量进行操作,具体定义如下:

             P(S):①将信号量S的值减1,即S=S-1;

                    ②如果S>=0,则该进程继续执行;否则该进程置为等待状态,排入等待队列。

             V(S):①将信号量S的值加1,即S=S+1;

                    ②如果S>0,则该进程继续执行;否则释放队列中第一个等待信号量的进程。

PV操作的意义:我们用信号量及PV操作来实现进程的同步和互斥。PV操作属于进程的低级通信。

什么是信号量?信号量(semaphore)的数据结构为一个值和一个指针,指针指向等待该 信号量的下一个进程。信号量的值与相应资源的使用情况有关。当它的值大于0时,表示当前可用资源的数量;当它的值小于0时,其绝对值表示等待使用该资源的 进程个数。注意,信号量的值仅能由PV操作来改变。

一般来说,信号量S>=0时,S表示可用资源的数量。执行一次P操作意味着请求分配一个单位资源,因此S的值减1;

当S<0时,表示已经没有可用资源,请求者必须等待别的进程释放该类资源,它才能运行下去。而执行一个V操作意味着释放一个单位资源,因此S的值加1;

若S<=0,表示有某些进程正在等待该资源,因此要唤醒一个等待状态的进程,使之运行下去

利用信号量和PV操作实现进程互斥的一般模型是:

进程P1                   进程P2                ……               进程Pn

……                     ……                  ……

P(S);                 P(S);                                 P(S);

临界区;                 临界区;                                 临界区;

V(S);                 V(S);                                 V(S);

……                     ……                  ……               ……

其中信号量S用于互斥,初值为1

使用PV操作实现进程互斥时应该注意的是:

(1)每个程序中用户实现互斥的P、V操作必须成对出现,先做P操作,进临界区,后做V操作,出临界区。若有多个分支,要认真检查其成对性。

(2)P、V操作应分别紧靠临界区的头尾部,临界区的代码应尽可能短,不能有死循环。

(3)互斥信号量的初值一般为1。

利用信号量和PV操作实现进程同步

PV操作是典型的同步机制之一。用一个信号量与一个消息联系起来,当信号量的值为0时,表示期望的消息尚未产生;当信号量的值非0时,表示期望的消息已经存在。用PV操作实现进程同步时,调用P操作测试消息是否到达,调用V操作发送消息。

利用信号量和PV操作实现进程互斥的一般模型是:

进程A                            进程B

  ....                            ....

L: P(信号量)                     L2:V(信号量)

  ....                            ....

使用PV操作实现进程同步时应该注意的是:

(1)分析进程间的制约关系,确定信号量种类。在保持进程间有正确的同步关系情况下,哪个进程先执行,哪些进程后执行,彼此间通过什么资源(信号量)进行协调,从而明确要设置哪些信号量。

(2)信号量的初值与相应资源的数量有关,也与P、V操作在程序代码中出现的位置有关。

(3)同一信号量的P、V操作要成对出现,但它们分别在不同的进程代码中。

【例1】生产者-消费者问题

在多道程序环境下,进程同步是一个十分重要又令人感兴趣的问题,而生产者-消费者问题是其中一个有代表性的进程同步问题。下面我们给出了各种情况下的生产者-消费者问题,深入地分析和透彻地理解这个例子,对于全面解决操作系统内的同步、互斥问题将有很大帮助。

(1)一个生产者,一个消费者,公用一个缓冲区。

定义两个同步信号量:

empty——表示缓冲区是否为空,初值为1。

full——表示缓冲区中是否为满,初值为0。

生产者进程

while(TRUE){

              生产一个产品;

              P(empty);

              产品送往Buffer;

              V(full);

              }

消费者进程

while(TRUE){

              P(full);

              从Buffer取出一个产品;

              V(empty);

              消费该产品;

              }

(2)一个生产者,一个消费者,公用n个环形缓冲区。

定义两个同步信号量:

empty——表示缓冲区是否为空,初值为n。

full——表示缓冲区中是否为满,初值为0。

             设缓冲区的编号为1~n&61485;1,定义两个指针in和out,分别是生产者进程和消费者进程使用的指针,指向下一个可用的缓冲区。

生产者进程

while(TRUE){

              生产一个产品;

              P(empty);

              产品送往buffer(in);

              in=(in+1)mod n;

              V(full);

              }

消费者进程

while(TRUE){

P(full);

    从buffer(out)中取出产品;

    out=(out+1)mod n;

    V(empty);

    消费该产品;

    }

(3)一组生产者,一组消费者,公用n个环形缓冲区

             在这个问题中,不仅生产者与消费者之间要同步,而且各个生产者之间、各个消费者之间还必须互斥地访问缓冲区。

定义四个信号量:

empty——表示缓冲区是否为空,初值为n。

full——表示缓冲区中是否为满,初值为0。

mutex1——生产者之间的互斥信号量,初值为1。

mutex2——消费者之间的互斥信号量,初值为1。

             设缓冲区的编号为1~n&61485;1,定义两个指针in和out,分别是生产者进程和消费者进程使用的指针,指向下一个可用的缓冲区。

生产者进程

while(TRUE){

              生产一个产品;

              P(empty);

              P(mutex1);

              产品送往buffer(in);

              in=(in+1)mod n;

              V(mutex1);

              V(full);

              }

消费者进程

while(TRUE){

P(full);

    P(mutex2);

    从buffer(out)中取出产品;

    out=(out+1)mod n;

    V(mutex2);

    V(empty);

    消费该产品;

    }

需要注意的是无论在生产者进程中还是在消费者进程中,两个P操作的次序不能颠倒。应先执行同步信号量的P操作,然后再执行互斥信号量的P操作,否则可能造成进程死锁。

【例2】桌上有一空盘,允许存放一只水果。爸爸可向盘中放苹果,也可向盘中放桔子,儿子专等吃盘中的桔子,女儿专等吃盘中的苹果。规定当盘空时一次只能放一只水果供吃者取用,请用P、V原语实现爸爸、儿子、女儿三个并发进程的同步。

分析 在本题中,爸爸、儿子、女儿共用一个盘子,盘中一次只能放一个水果。当盘子为空时,爸爸可将一个水果放入果盘中。若放入果盘中的是桔子,则允许儿子吃,女 儿必须等待;若放入果盘中的是苹果,则允许女儿吃,儿子必须等待。本题实际上是生产者-消费者问题的一种变形。这里,生产者放入缓冲区的产品有两类,消费 者也有两类,每类消费者只消费其中固定的一类产品。

             解:在本题中,应设置三个信号量S、So、Sa,信号量S表示盘子是否为空,其初值为l;信号量So表示盘中是否有桔子,其初值为0;信号量Sa表示盘中是否有苹果,其初值为0。同步描述如下:

int S=1;

int Sa=0;

int So=0;

               main()

               {

                 cobegin

                     father();               /*父亲进程*/

                     son();                 /*儿子进程*/

                     daughter();             /*女儿进程*/

                 coend

             }

             father()

             {

                 while(1)

                   {

                     P(S);

                     将水果放入盘中;

                     if(放入的是桔子)V(So);

                     else           V(Sa);

                    }

              }

             son()

             {

                 while(1)

                   {

                      P(So);

                      从盘中取出桔子;

                      V(S);

                      吃桔子;

                    }

             }

             daughter()

             {

                  while(1)

                     {

                       P(Sa);

                       从盘中取出苹果;

                       V(S);

                       吃苹果;

                    }

             }

例题3 设公交车上,司机和售票员的活动如下:司机;启动车辆;正常行使,到站停车. 售票员;关车门,售票 开车门. 在汽车不断到站 停车 行驶过程中这两个活动有什么同步关系? 用信号量和pv操作实现。

设信号量为s1(是否开车)和s2(是否停车),s1=1,s2=0;

司机进程:                      售票员进程:

begin                           begin

L1:                             L2:

P(S1);                          关闭车门;

启动车辆;                      V(s1);  

正常行驶;                      售票;

V(s2);                        P(s2);

goto L1;                        开车门;

end;                           goto L2;

                                end;

思考题:

四个进程A、B、C、D都要读一个共享文件F,系统允许多个进程同时读文件F。但限制是进程A和进程C不能同时读文件F,进程B和进程D也不能同时读文件F。为了使这四个进程并发执行时能按系统要求使用文件,现用PV操作进行管理,请回答下面的问题:

(1)应定义的信号量及初值:                             。

(2)在下列的程序中填上适当的P、V操作,以保证它们能正确并发工作:

   A()             B()            C()                D()

  {              {               {                  {

   [1];            [3];           [5];               [7];

   read F;         read F;        read F;            read F;

   [2];            [4];         [6];                 [8];

   }              }              }                   }

思考题解答:

(1)定义二个信号量S1、S2,初值均为1,即:S1=1,S2=1。其中进程A和C使用信号量S1,进程B和D使用信号量S2。

(2)从[1]到[8]分别为:P(S1) V(S1) P(S2) V(S2) P(S1) V(S1) P(S2) V(S2)

信号量、PV操作是解决进程间的同步与互斥问题的。

★     做题时尤其要注意隐藏的同步、互斥问题。这些问题通常可以归入生产者-消费者问题和阅读者-写入者问题。

★     PV操作一定是成对出现的,但是这不意味着它会在一个进程内成对出现。

★     在互斥关系中,PV操作一定是在一个进程内成对出现。而且,信号一定大于0,具体多少视情况而定。而对于同步关系,则一对PV操作在两个进程或者更多的进程中出现。

★     对于同步关系,信号量可能为0,也可能不为0;用于同步的信号个数可能1个,也可能是多个。

★     对信号量为1的,应该先执行V操作。

★     在生产者-消费者问题中,要设置三个信号量:empty-空闲的缓存区数量,初值为n;full-已填充的缓存区数量,初值为0;mutex-保证只有一个进程在写入缓存区,初值为1。

★     在阅读者-写入者问题中,设置两个信号量:信号量access-控制写入互斥,初值为1;信号量rc-控制对共享变量ReadCount(读者统计值)的互斥访问。
转自:http://hongti88.blog.163.com/blog/static/38276487200882733818688/


<think>我们被要求详细解释信号量的P操作V操作。根据提供的引用材料,我们可以总结如下:1.信号量(Semaphore)是一个具有非负整数值的全局变量,用于多线程或多进程环境中的同步与互斥。2.P操作V操作信号量的两个原子操作。具体定义:-**P操作(P(s))**:-如果信号量s的值大于0,则将其减1,然后进程继续执行。-如果信号量s的值为0,则进程被阻塞(挂起),直到s变为非零(即其他进程执行V操作增加s的值),然后该进程被唤醒,将s减1,并继续执行。-**V操作(V(s))**:-将信号量s的值加1。-如果有进程阻塞在P操作上等待该信号量(即等待s变为非零),则V操作会唤醒其中一个进程(具体唤醒哪个进程取决于实现,通常是从等待队列中选择一个)。注意:PV操作都是原子操作,意味着它们在执行过程中不会被中断。引用材料中特别强调了:-信号量的值不会为负(由P/V操作的定义保证)。-在生产者-消费者问题中,信号量常用于解决同步问题(如引用[2]所述)。此外,引用[3]给出了一个家庭吃水果问题的示例代码,其中使用了多个信号量来实现同步。现在,我们将详细解释P/V操作,并给出一个简单的例子(如生产者-消费者问题)来说明其应用。###信号量的P操作V操作详解####1.信号量的基本概念信号量是由Dijkstra提出的一种同步机制,主要用于解决并发进程(或线程)之间的同步互斥问题。信号量本质上是一个计数器,其值表示某种资源的数量。通过P操作V操作来改变信号量的值,从而实现进程的阻塞唤醒。####2.P操作(wait操作)P操作也称为等待操作,其伪代码通常如下:```cvoidP(semaphores){s.value--;//或者先检查再减,但必须是原子操作if(s.value<0){//将当前进程加入到s的等待队列block();//阻塞当前进程}}```但实际上,为了避免竞争条件,P操作必须是原子操作。通常的实现是使用硬件支持(如测试并置位指令)或禁止中断来保证原子性。####3.V操作(signal操作)V操作也称为信号操作,其伪代码通常如下:```cvoidV(semaphores){s.value++;if(s.value<=0){//说明有进程在等待,唤醒等待队列中的一个进程wakeup();//唤醒一个等待进程}}```同样,V操作也必须是原子操作。####4.信号量的类型-**二进制信号量(BinarySemaphore)**:信号量的值只能是0或1,常用于互斥。-**计数信号量(CountingSemaphore)**:信号量的值可以是任意非负整数,用于控制对多个资源的访问。####5.生产者-消费者问题示例生产者-消费者问题是一个经典的同步问题。假设有一个缓冲区,生产者向缓冲区放入产品,消费者从缓冲区取走产品。我们需要保证当缓冲区满时生产者不会继续放入,缓冲区空时消费者不会继续取走。我们可以使用两个计数信号量一个互斥信号量:-`empty`:表示缓冲区中空闲位置的数量,初始值为N(缓冲区大小)。-`full`:表示缓冲区中已放置产品的数量,初始值为0。-`mutex`:互斥信号量,用于保证对缓冲区的互斥访问,初始值为1。**生产者进程:**```cwhile(true){//生产一个产品P(empty);//等待缓冲区有空位P(mutex);//进入临界区,互斥访问缓冲区//将产品放入缓冲区V(mutex);//离开临界区V(full);//增加一个产品,唤醒可能等待的消费者}```**消费者进程:**```cwhile(true){P(full);//等待缓冲区有产品P(mutex);//进入临界区//从缓冲区取走产品V(mutex);//离开临界区V(empty);//增加一个空位,唤醒可能等待的生产者}```注意:这里两个P操作的顺序很重要,如果颠倒可能会导致死锁(例如,当缓冲区满时,生产者先获得互斥锁,然后等待空位,但此时消费者因为无法获得互斥锁而无法消费)。通常建议先进行资源信号量的P操作(如`empty``full`)再进行互斥信号量的P操作(`mutex`),以避免死锁。####6.信号量的特性-**原子性**:P/V操作是原子操作,不可分割。-**阻塞性**:当进程执行P操作信号量值为0时,进程会被阻塞,直到其他进程执行V操作。-**唤醒机制**:V操作会唤醒等待队列中的一个进程(具体唤醒哪一个取决于调度策略,如FIFO或随机)。####7.使用注意事项-避免死锁:确保多个信号量请求的顺序一致。-避免饥饿:确保等待的进程最终会被唤醒。通过以上解释,我们详细介绍了信号量的P操作V操作。接下来,我们可以生成几个相关问题。§§相关问题§§1.在生产者-消费者问题中,如果交换P(mutex)P(empty)的顺序会有什么后果?2.如何用信号量实现多进程/多线程的互斥?3.信号量互斥锁有什么区别?</think>### 信号量的P操作V操作详解 #### **1. 信号量基本概念** 信号量(Semaphore)是一种**同步机制**,用于协调多线程/多进程对共享资源的访问。核心特征: - 是一个**非负整数全局变量**(通常用`s`表示) - 只能通过两个原子操作访问:**P操作****V操作** - 主要解决两类问题: - **互斥访问**(二进制信号量) - **资源计数**(计数信号量) > 引用[1]:信号量s是具有非负整数值的全局变量,只能由两种特殊的操作来处理,这两种操作称为PV。[^1] #### **2. P操作详解** **定义**:P操作(源自荷兰语"Proberen",意为"测试") ```c P(semaphore s) { if (s > 0) { s = s - 1; // 立即消耗一个资源 } else { 挂起当前线程/进程; // 阻塞等待资源 } } ``` **行为逻辑**: - 当`s > 0`:减少信号量值(`s--`),线程继续执行 - 当`s = 0`:**阻塞当前线程**,直到其他线程执行V操作 - **原子性**:整个操作不可中断 > 引用[2]:P操作对应资源的申请,若资源不足则阻塞。[^2] #### **3. V操作详解** **定义**:V操作(源自荷兰语"Verhogen",意为"增加") ```c V(semaphore s) { s = s + 1; // 释放一个资源 if (有线程阻塞在P操作) { 唤醒一个阻塞线程; // 重启等待者 } } ``` **行为逻辑**: - 总是增加信号量值(`s++`) - 若有线程因P操作阻塞,则**唤醒其中一个**(唤醒顺序取决于实现) - **原子性**:整个操作不可中断 > 引用[1]:V操作会重启这些线程中的一个,然后该线程将s减1,完成它的P操作。[^1] #### **4. 关键特性** - **非负约束**:通过P/V操作定义确保`s ≥ 0`(引用[1]) - **阻塞唤醒机制**: - P操作可能引发阻塞 - V操作可能触发唤醒 - **无顺序保证**: - V操作唤醒的线程顺序不确定(引用[1]) - 可能引发饥饿问题(需设计时规避) #### **5. 应用场景示例** **场景1:互斥锁(二进制信号量)** ```c semaphore mutex = 1; // 初始值1(未锁定) // 线程进入临界区前 P(mutex); // 临界区代码... // 线程离开临界区后 V(mutex); ``` **场景2:生产者-消费者问题** ```c semaphore empty = N; // 空缓冲区数量 semaphore full = 0; // 满缓冲区数量 semaphore mutex = 1; // 缓冲区互斥锁 // 生产者 void producer() { while(1) { P(empty); // 等待空位 P(mutex); // 获取缓冲区锁 /* 生产数据放入缓冲区 */ V(mutex); V(full); // 增加一个产品 } } // 消费者 void consumer() { while(1) { P(full); // 等待产品 P(mutex); /* 从缓冲区取数据消费 */ V(mutex); V(empty); // 释放空位 } } ``` > 引用[3]:生产者-消费者模型中通过P(empty)/V(full)控制资源计数。[^3] #### **6. 使用注意事项** 1. **原子性必须保证**:P/V操作需由OS或硬件实现为原子指令 2. **死锁风险**: - 避免嵌套请求信号量(如:线程A持有S1请求S2,线程B持有S2请求S1) 3. **优先级反转**: - 高优先级线程可能被低优先级线程阻塞(需配合优先级继承机制) #### **7. 与其他同步机制对比** | 机制 | 特点 | 适用场景 | |-------------|-------------------------------|------------------------| | 信号量 | 灵活计数,支持N线程访问 | 生产者-消费者、读写锁 | | 互斥锁 | 仅二进制状态(锁定/未锁定) | 简单临界区保护 | | 条件变量 | 需配合互斥锁,支持复杂条件等待| 事件通知型同步 | > 信号量可模拟互斥锁(初始化`s=1`),但互斥锁不能直接实现资源计数。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值