信号量实现同步之司机售票员问题

1、信号量是什么

信号量是一种进程同步工具,可以同步并发进程,相较于互斥锁,可以解决更多类型的同步问题(同步即是指进程之间有序执行,而异步是指随机执行)。

2、信号量如何实现

信号量是一个整数值,除了初始化其他时间内只能通过PV操作改变其值。信号量通过两个操作来实现,分别命名为P操作和V操作。其中P操作是指等待(wait operation),V操作是指信号数量增加(signal operation)。这两种操作均为原子操作。

p(s){
    while(s<=0)
        do nothing;
    s--;
}

v(s){
    s++;
}

3、信号量不同取值的应用场景

(1)信号量取值为0或1用于实现互斥锁的作用

semaphore mutex = 1;
 process pi{
    p(mutex);

    critical section

    v(mutex);
}

(2)信号量取值为大于1,一般信号量可以取任意值,可以控制并发进程对共享资源的访问,用于表示可控资源的数量。

semaphore road = 2;

process Carsi{
    p(road);

    pass the fork in the road

    v(road);
}

(3)初始值设置为0则用于进程同步。

4、信号量实现同步实例:司机售票员

司机:启动车辆->正常行车->到站停车

售票员:关车门->售票->开车门

规则:司机要等车门关闭才能开车 售票员要等车停下才能开门

//
//  driver_conductor.c
//  
//
//  Created by YIN on 2021/5/16.
//

#include "driver_conductor.h"
#include <stdio.h>
#include <pthread.h>
#include <semaphore.h>
#include <unistd.h>
#include <dispatch/dispatch.h>

dispatch_semaphore_t d;
dispatch_semaphore_t c;

/*
 司机:启动车辆;正常行车;到站停车。
 售票:关车门;售票;开车门。
 */

void* driver(void* args)
{
    dispatch_semaphore_wait(d, DISPATCH_TIME_FOREVER);
    printf("Start the vehicle\n");
    printf("Normal driving\n");
    printf("Stop at the station\n");
    dispatch_semaphore_signal(c);
}

void* conductor(void* args)
{
    printf("Close the car door\n");
    dispatch_semaphore_signal(d);
    printf("Ticket sales\n");
    dispatch_semaphore_wait(c, DISPATCH_TIME_FOREVER);
    printf("Open the car door\n");
}


int main(int argc, char const *argv[])
{
    pthread_t pid_driver;
    pthread_t pid_conductor;
    
    dispatch_semaphore_t *sem_d = &d;
    *sem_d = dispatch_semaphore_create(0);
    
    dispatch_semaphore_t *sem_c = &c;
    *sem_c = dispatch_semaphore_create(0);
    
    pthread_create(&pid_driver, NULL, driver, NULL);
    pthread_create(&pid_conductor, NULL, conductor, NULL);
   
    pthread_join(pid_driver, NULL);
    pthread_join(pid_conductor, NULL);
    
    return 0;
}

 

 

  • 1
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
假设有一个公交车的售票员司机需要进行同步操作,售票员需要在每一站点停车后向乘客售票,而司机则需要等待售票员售票完成后才能继续行驶。 我们可以使用线程和信号量机制来实现这个同步操作问题。具体实现步骤如下: 1. 创建两个线程,一个线程代表售票员,一个线程代表司机。 2. 创建一个信号量,初始值为0,用于表示售票员是否已经完成售票操作。 3. 在售票员线程中,售票员首先等待司机发出停车信号。一旦司机发出停车信号,售票员开始售票操作,售票员售票完成后释放信号量。 4. 在司机线程中,司机首先行驶到下一站点,然后发出停车信号,等待售票员完成售票操作,一旦售票员完成售票操作,司机释放售票信号量并继续行驶。 代码示例: ```python import threading ticket_semaphore = threading.Semaphore(0) def driver(): while True: # 行驶到下一站点 print("Driver: Next stop.") # 发出停车信号 ticket_semaphore.release() # 等待售票员完成售票操作 ticket_semaphore.acquire() # 继续行驶 print("Driver: Continue driving.") def conductor(): while True: # 等待司机发出停车信号 ticket_semaphore.acquire() # 售票员开始售票操作 print("Conductor: Selling tickets.") # 售票员售票完成 ticket_semaphore.release() driver_thread = threading.Thread(target=driver) conductor_thread = threading.Thread(target=conductor) driver_thread.start() conductor_thread.start() ``` 在这个示例中,售票员司机使用一个信号量进行同步操作,售票员在完成售票操作后释放信号量司机在发出停车信号后等待售票员完成售票操作并释放信号量。这样就可以保证售票员司机的操作是同步的。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值