操作系统-进程同步(二)

进程同步:synchronization 指系统中多个进程中发生的事件存在某种时序关系,需要相互合作,共同完成一项任务。具体地说,一个进程运行到某一点时,要求另一伙伴进程为它提供消息,在未获得消息之前,该进程进入阻塞态,获得消息后被唤醒进入就绪态。

生产者/消费者问题

问题描述:一个或多个生产者生产某种类型的数据放置在缓冲区中,有消费者从缓冲区中取数据,每次取一项,只能有一个生产者或消费者对缓冲区进行操作。
要解决的问题:当缓冲区已满时,生产者不会继续向其中添加数据;当缓冲区为空时,消费者不会从中移走数据
避免忙等待:睡眠或唤醒原语

#define N 100
int count=0;
void producer(void)
{  int item;
    while(TRUE){
      item=produce_item();
      if(count==N) sleep();
      insert_item(item);
      count=count+1;
      if(count==1)
        wakeup(consumer);
    }
}

void consumer(void)
{
  int item;
  while(TRUE){
    if(count==0) sleep();
    item=remove_item();
    count=count-1;
    if(count==N-1)
      wakeup(producer);
    consume_item(item);
     }
}

信号量及P、V操作

信号量和PV操作是一种卓有成效的进程同步机制,1965年由荷兰学者Dijkstra提出。
一个特殊变量,用于进程间传递信息的一个整数值,定义如下

struc semaphore
{
  int count;
  queueType queue;
}

信号量说明:semaphore s;
对信号量可以实施的操作:初始化、P和V(P、V分别是荷兰语的test(proberen)和increment(verhogen)
P、V操作定义

P(s)
{
  s.count--;
  if(s.count<0)
  {
    该进程状态置为阻塞状态;
    该进程插入相应的等待队列s.queue末尾;
    重新调度;
  }
}

V(s)
{
  s.count++;
  if(s.count<=0)
  {
    唤醒相应等待队列s.queue中等待的一个进程;
    改变其状态为就绪态,并将其插入就绪队列;
  }
}

P、V操作为原语操作
在信号量上定义了三个操作:初始化(非负数)、P操作、V操作
最初提出的是二元信号量(解决互斥),之后,推广到一般信号量(多值)或计数信号量(解决同步)

用PV操作解决进程间互斥问题

  • 分析并发进程的关键活动,划定临界区
  • 设置信号量mutex,初值为1
  • 在临界区前实施P(mutex)
  • 在临界区后实施V(mutex)

用信号量解决生产者、消费者问题

操作系统-进程同步(二)_第1张图片
Paste_Image.png
操作系统-进程同步(二)_第2张图片
Paste_Image.png

读者写者问题

用信号量解决读者/写者问题
问题描述: 多个进程共享一个数据区,这些进程分为两组:
读者进程:只读数据区中的数据
写者进程:只往数据区写数据
要求满足条件:

  • 允许多个读者同时执行读操作
  • 不允许多个写者同时操作
  • 不允许读者、写者同时操作

第一类读者写者问题:读者优先

如果读者执行:

  • 无其他读者、写者。该读者可以读
  • 若已有写者等,但有其他读者正在读,则该读者也可以读
  • 若有写者正在写,该读者必须等

如果写者执行:

  • 无其他读者、写者,该写者可以写
  • 若有读者正在读,该写者等待
  • 若有其他写者正在写,该写者等待
操作系统-进程同步(二)_第3张图片
Paste_Image.png

Linux提供的读写锁

应用场景:如果每个执行实体对临界区的访问或者是读或者是写共享变量,但是它们都不会既读又写时,读写锁是最好的选择
实例:Linux的IPX路由代码中使用了读写锁,用ipx_routes_lock的读写锁保护IPX路由表的并发访问
要通过查找路由表实现包转发的程序需要请求读锁;需要添加和删除路由表中入口的程序必须获取写锁(由于通过读路由表的情况比更新路由表的情况多得多,使用读写锁提高了性能)

你可能感兴趣的:(操作系统-进程同步(二))