16.信号量概述

1.信号量

信号量(semaphore)与已经介绍过的 IPC 结构不同,它是一个计数器。信号量用于实现进程间的互斥与同步,而不是用于存储进程间通信数据。

特点:

         (1)信号量用于进程间同步,若要在进程间传递数据需要结合共享内存

         (2)信号量基于操作系统的 PV 操作,程序对信号量的操作都是原子操作。

         (3)每次对信号量的 PV 操作不仅限于对信号量值加 1 或减 1,而且可以加减任意正整数。

        (4)支持信号量组。

2、原型

最简单的信号量是只能取 0 和 1 的变量,这也是信号量最常见的一种形式,叫做二值信号量(Binary Semaphore)。而可以取多个正整数的信号量被称为通用信号量。

Linux 下的信号量函数都是在通用的信号量数组上进行操作,而不是在一个单一的二值信号量上进行操作。

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>

// 创建或获取一个信号量组:若成功返回信号量集ID,失败返回-1
int semget(key_t key, int num_sems, int sem_flags);

// 对信号量组进行操作,改变信号量的值:成功返回0,失败返回-1
int semop(int semid, struct sembuf semoparray[], size_t numops);

// 控制信号量的相关信息
int semctl(int semid, int sem_num, int cmd, ...);

semget创建新的信号量集合时,必须指定集合中信号量的个数(即num_sems),通常为1; 如果是引用一个现有的集合,则将num_sems指定为 0 。 

示例代码:

sem.c

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include<stdio.h>


union semun {
       int              val;    /* Value for SETVAL */
       struct semid_ds *buf;    /* Buffer for IPC_STAT, IPC_SET */
       unsigned short  *array;  /* Array for GETALL, SETALL */
       struct seminfo  *__buf;  /* Buffer for IPC_INFO
                                           (Linux-specific) */
};

void pGetKey(int id)//取钥匙
{
      //int semop(int semid, struct sembuf *sops, unsigned nsops);

      struct sembuf set;

      set.sem_num = 0;
      set.sem_op = -1;//钥匙值减1

      set.sem_flg = SEM_UNDO;

      semop(id,&set,1);

      printf("getkey\n");
}

void vPutBackKey(int id)//归还钥匙
{
      //int semop(int semid, struct sembuf *sops, unsigned nsops);

      struct sembuf set;

      set.sem_num = 0;
      set.sem_op = 1;//钥匙值+1

      set.sem_flg = SEM_UNDO;

      semop(id,&set,1);

      printf("put back the key\n");
}

int main(int argc,char* argv[])
{
       //int semget(key_t key, int nsems, int semflg);

       int semid;
       key_t key;
       key = ftok(".",2);

                        //1信号量集合中有一个信号量
       semid = semget(key,1,IPC_CREAT|0666);//IPC_CREAT创建或获取信号量

       //int semctl(int semid, int semnum, int cmd, ...);

       union semun initsem;
       initsem.val = 0;//没有钥匙

                  //0为操作第0个信号量
       semctl(semid,0,SETVAL,initsem);//初始化信号量
                   //SETVAL设置信号量的值,设置为initsem

       int pid = fork();
       if(pid > 0)
       {
            pGetKey(semid);
            printf("this is father\n");
            vPutBackKey(semid);

            semctl(semid,0,IPC_RMID);//销毁钥匙
       }
       else if(pid == 0)
       {
            //pGetKey(semid);
            printf("this is child\n");
            vPutBackKey(semid);
       }
       else
       {
            printf("fork error\n");
       }
         return 0;
}

 

运行结果:

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值