Linux 进程间通信——信号量

一、信号量描述

信号量是一个特殊的变量,一般取正数值。它的值代表允许访问的资源数目。

获取资源时,需要对信号量的值进行原子减一,该操作被称为P操作,当信号量值为0时,代表没有资源可用,P操作会阻塞。释放资源时,需要对信号量的值进行原子加一,该操作被称为V操作。

信号量主要用来同步进程。

信号量的值如果只取0,1,将其称为二值信号量;如果信号量的值大于1,则称之为计数信号量。

临界资源:同一时刻,只允许被一个进程或线程访问的资源
临界区:访问临界资源的代码段

二、信号量使用

1.操作信号量的接口介绍

(1)semget()信号量集合的创建

semget 可以一次创建多个信号量,为了管理这些信号量,统一把这些信号量放在一个信号量集合中,如果创建成功,返回的就是信号集的ID

int semget(key_t  key, int nsems, int semflg); 

semget()创建或者获取已存在的信号量
semget()成功返回信号量的ID, 失败返回-1
key:两个进程使用相同的key值,就可以使用同一个信号量
nsems:内核维护的是一个信号量集,在新建信号量时,其指定信号量集中信号量的个数,如果不打算创建 一个新的信号量集合的时候,该参数可以是0。
semflg:信号量集合的访问权限,可选IPC_CREAT,IPC_EXCL。通常是IPC_CREAT | 0666,如果信号量集合已经存在,那么信号量集合的访问权限会被修改成当前填入的权限。

(2)semop()信号量的PV操作

semop包含了对信号量集合中信号量的 P操作(申请) 和 V操作(释放)。如果要一次操作多个信号量,那么就需要定义一个sembuf类型的结构体数组。

int semop(int semid, struct sembuf *sops, unsigned nsops);

semop()对信号量进行改变,做P操作或者V操作
semop()成功返回0,失败返回-1
semid:信号量集合的ID
struct sembuf *sops
输入型参数,要对信号量集合中的哪个信号量做什么操作,阻塞方式又是怎样。结构体sembuf 的声明如下:

struct sembuf  
{  
unsigned short sem_num;  //指定信号量集中的信号量下标  
short          sem_op;   //其值为-1,代表P操作,其值为1,代表V操作  
short          sem_flg;  //阻塞方式。0表示阻塞,IPC_NOWAIT表示非阻塞。当程序执行了p操作崩溃了之后,系统帮忙将该信号量释放掉
}; 

nsops:要操作的信号量的个数,一般是 1 。

(3)semctl()信号量的初始化 / 销毁

semctl 可以对信号量集合中的信号量进行初始化或者删除。因为现在只有信号量集合的ID,实际上我们要想操作信号量集合中的某一个信号量,需要先给这些信号量编号,然后根据编号来定向操作某一个信号量。

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

semctl()控制信号量的初始化 / 销毁
semctl()失败返回 -1,成功时会根据cmd的值返回不同的值(值是大于0的)
semid:信号量集合的ID
semnum

要操作信号量集合中的信号量编号。信号量集合给信号量编号的方式类似于数组下标,是从0开始的,所以为了区分哪个信号量是干什么用的,可以使用宏或者 枚举的方式来指明。如果第三个参数是IPC_RMID,当前参数会被无视。

union semun  
{  
int             val;  
struct semid_ds *buf;  
unsigned short  *array;  
struct seminfo  *_buf;  
}; 

cmd:要对信号量所作的操作。cmd的可选值如下:

在这里插入图片描述

2.封装信号量接口并在程序中使用信号量

例题:进程a和进程b模拟访问打印机,进程a输出第一个字符‘a’表示开始使用打印机,输出第二个字符‘a’表示结束使用,b进程操作与a进程相同。(由于打印机同一时刻只能被一个进程使用,所以输出结果不应该出现abab),如图所示:

在这里插入图片描述
(1)封装信号量的接口:

sem.h的代码如下:

在这里插入图片描述

sem.c的代码如下:

在这里插入图片描述
在这里插入图片描述

b.c的代码如下:

在这里插入图片描述

a.c的代码如下:

在这里插入图片描述

运行结果如下图所示:

在这里插入图片描述

【注意】信号量的创建和销毁只有一次就好了。如果信号量没有被创建,那么创建一个信号量,如果信号量被创建了,就获取也就创建的信号量。信号量的销毁也一样,仅销毁一次就行了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
信号量是一种用于进程间通信和同步的机制。它是一个计数器,用于保证在共享资源上的互斥访问。在Linux系统中,可以使用信号量来实现进程间的同步和互斥。以下是信号量的基本概念: - 计数器:信号量的值是一个计数器,它可以被多个进程共享。 - P操作:当一个进程需要访问共享资源时,它必须执行P操作,该操作会将信号量的值减1。如果信号量的值为0,则进程将被阻塞,直到信号量的值大于0。 - V操作:当一个进程使用完共享资源后,它必须执行V操作,该操作会将信号量的值加1。如果有进程正在等待该信号量,则唤醒其中一个进程继续执行。 在ZUCC中,可以使用信号量来实现进程的同步和互斥。首先,需要使用semget函数创建一个信号量集合,并使用semctl函数对信号量进行初始化。然后,可以使用semop函数执行P和V操作。例如,下面是一个简单的示例程序,用于演示信号量的使用: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/sem.h> #define SEM_KEY 1234 union semun { int val; struct semid_ds *buf; unsigned short *array; }; int main() { int semid, pid; union semun arg; struct sembuf sb; // 创建信号量集合 semid = semget(SEM_KEY, 1, IPC_CREAT | 0666); if (semid == -1) { perror("semget"); exit(EXIT_FAILURE); } // 初始化信号量 arg.val = 1; if (semctl(semid, 0, SETVAL, arg) == -1) { perror("semctl"); exit(EXIT_FAILURE); } // 创建子进程 pid = fork(); if (pid == -1) { perror("fork"); exit(EXIT_FAILURE); } else if (pid == 0) { // 子进程执行P操作 sb.sem_num = 0; sb.sem_op = -1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop P"); exit(EXIT_FAILURE); } printf("Child process\n"); // 子进程执行V操作 sb.sem_num = 0; sb.sem_op = 1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop V"); exit(EXIT_FAILURE); } exit(EXIT_SUCCESS); } else { // 父进程执行P操作 sb.sem_num = 0; sb.sem_op = -1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop P"); exit(EXIT_FAILURE); } printf("Parent process\n"); // 父进程执行V操作 sb.sem_num = 0; sb.sem_op = 1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop V"); exit(EXIT_FAILURE); } exit(EXIT_SUCCESS); } return 0; } ``` 在上述代码中,创建了一个信号量集合,并将其初始化为1。然后,创建了一个子进程和一个父进程,它们分别执行P和V操作。由于信号量的初始值为1,因此父进程和子进程都可以顺利地执行。如果将信号量的初始值改为0,那么父进程和子进程都将被阻塞,直到有一个进程执行V操作为止。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值