IPC——信号量

信号量的基本概念

信号量(又称信号灯)与其它进程间通信方式不大相同,主要用途是保护临近资源,进程可以根据它判断是否能否访问某些共享资源,除了用于访问控制外,还可以进行进程同步。   

 分类

二值信号灯:信号灯的值只能取零和一,类似于互质锁,但两者又不同:信号灯强调共享资源,只要共享资源可用,其它进程同样可以修改信号灯的值;互斥锁则更强调进程,占用资源的进程使用完毕后,必须由进程本身来解锁。

 

计数信号灯:信号灯的值可以取任意非负值

 

创建或者打开信号量:

#include <sys/types.h>

#include <sys/ipc.h>

#include <sys/sem.h>

 

int  semget(key_t key, int nsems , int  ****)

key: 键值,由ftok获得

nsems:指定打开或者创建新的信号灯集中将包含的信号灯的数目

****:标识,同消息队列

 

对信号量的操作:

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

功能:对信号量进行控制

semid:信号量集的ID

sops:是一个操作数组,表明要进行什么操作

nsopssops所指向的数组的元素个数

 

struct sembuf{

unsigned shor sem_num;

short sem_op;

short  **;

}

 **标识可选择的有两种:

IPC_NOWAIT:对信号的操作不能满足时,semop()不会被阻塞,它会立即返回,同时设定错误信息。

IPC_UNDO :程序结束时(无论正常或者不正常)释放信号量,这样做的目的是在于避免程序在异常的情况下结束时未能将锁定的资源解锁,造成该资源永远锁定。
信号量是一种用于进程间通信和同步的机制。它是一个计数器,用于保证在共享资源上的互斥访问。在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、付费专栏及课程。

余额充值