封装p操作和v操作(拿锁,放回锁)
让父进程先拿锁
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/sem.h>
// int semget(key_t key, int nsems, int semflg);
//int semctl(int semid, int semnum, int cmd, ...);
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) */
};
// int semop(int semid, struct sembuf *sops, size_t nsops);
void pgetkey(int id)//拿锁 man 2 semop
{
struct sembuf set;//定义结构体变量
set.sem_num=0;//信号量的编号
set.sem_op=-1;//把钥匙的值减一代表拿钥匙
set.sem_flg=SEM_UNDO;//拿不到锁,等待
semop(id,&set,1);
printf("getkey\n");
}
void vputbackkey(int id)//放回锁
{
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)
{
key_t key;
int semid;
key=ftok(".",2);
semid=semget(key,1,IPC_CREAT|0666);//创建信号量
// 1 信号量中只有一个信号量
union semun initsem;
initsem.val=0;//只有一把锁
semctl(semid,0,SETVAL,initsem);//初始化信号量
//0 操作第0个信号量
//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)
{
printf("this is child\n");
vputbackkey(semid);
}
else
{
printf("fork error\n");
}
return 0;
}
运行结果:
p操作和v操作的知识点总结:
P操作:
若信号量值为1,获取资源并将信号量值变为-1。
若信号量值为0,进程挂起等待
V操作
释放资源并将信号量值+1。
如果有进程正在挂起等待,则唤醒它们
——@上官可编程