Linux--信号量

一、信号量
信号量 (semaphore)与已经介绍过的IPC结构不同,他是一个计数器。

用于实现进程间的互斥与同步,而不是用于存储进程间的通信数据。

可以与共享内存配合使用。
临界资源: 多道程序系统种存在许多进程,他们共享各种资源,然而有很多资源一次智能供一个进程使用。一次仅允许一个进程使用 的资源称为临界资源。许多物理设备都属于临界资源,如输入机、打印机、磁带机等。
信号量集: 相当于信号量的集合。
P V 操作 P拿取,V放回

1、特点:

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

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

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

④、支持信号量组

2、原型:

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

c 复制代码
#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); 
//3.控制信号量的相关信息
int semctl(int semid, int sem_num, int cmd, ...);

3、例:

c 复制代码
//信号量

/*
#include <sys/types.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, ...);

struct sembuf
{
    short sem_num;
    short sem_op;
    short sem_flg;
}

*/

#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)*/
};

int main(int argc,char const *argv[])
{
	key_t key;
	int semid;
	
	key = ftok(".",2);
	
	semid = semget(key,1,IPC_CREAT|0666);
	//获取信号量(信号量中有一个合集)
	
	//定义一个联合体
	union semun initsem;
	initsem.val = 1;
	
	semctl(semid,0,SATVAL,initsem);
	//做初始化信号量,操作第0个信号量,SATVAL设置信号量的值为initsem,


	int pid = fork();
	if(pid > 0){
		printf("this is father\n");
	}
	else if(pid == 0){
		printf("this is child\n");
	}
	else{
		printf("fork error\n");
	}
	return 0;
}

此程序运行一般为父进程先运行,然后运行子进程。偶尔反转。

修改程序,使信号量控制让子进程先运行:

c 复制代码
//修改程序,使信号量控制让子进程先运行:封装两个函数 P操作和 V操作

/*
#include <sys/types.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, ...);
//		 int semop(int semid, struct sembuf *sops, unsigned nsops); 
struct sembuf short;
{
    short.sem_num;
    short.sem_op;
    short.sem_flg;
}

*/

#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)*/
};

//P操作
void pGetKey(int id)
{
	struct sembuf set;

    set.sem_num = 0;
    set.sem_op = -1;
    set.sem_flg = SEM_UNDO;
	semop(id,&set,1);
	printf("get ket!\n");
}


//V操作
void vPutBackKey(int id)
{
	struct sembuf set;

    set.sem_num = 0;
    set.sem_op = -1;
    set.sem_flg = SEM_UNDO;
	semop(id,&set,1);
	printf("put back ket!\n");
}

int main(int argc,char const *argv[])
{
	key_t key;
	int semid;
	
	key = ftok(".",2);
	
	semid = semget(key,1,IPC_CREAT|0666);
	//获取信号量(信号量中有一个合集)
	
	//定义一个联合体
	union semun initsem;
	initsem.val = 0;
	
	semctl(semid,0,SATVAL,initsem);
	//做初始化信号量,操作第0个信号量,SATVAL设置信号量的值为initsem,


	int pid = fork();
	if(pid > 0){
		pGetKey(ssemid);
		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;
}
相关推荐
web_1553427465622 分钟前
性能巅峰对决:Rust vs C++ —— 速度、安全与权衡的艺术
c++·算法·rust
9毫米的幻想24 分钟前
【Linux系统】—— 冯诺依曼体系结构与操作系统初理解
linux·运维·服务器·c语言·c++
刘什么洋啊Zz31 分钟前
剖析IO原理和零拷贝机制
java·运维·网络
升讯威在线客服系统1 小时前
如何通过 Docker 在没有域名的情况下快速上线客服系统
java·运维·前端·python·docker·容器·.net
leoufung2 小时前
vim 多个关键字高亮插件介绍
linux·编辑器·vim
Karoku0663 小时前
【CI/CD】CI/CD环境搭建流程和持续集成环境配置
运维·ci/cd·docker·容器·kubernetes·prometheus
Nerd Nirvana5 小时前
软考—系统架构设计(案例 | 论文)
linux·系统架构·软件工程·软考·计算机基础
勤奋的凯尔森同学6 小时前
webmin配置终端显示样式,模仿UbuntuDesktop终端
linux·运维·服务器·ubuntu·webmin
计算机小白一个7 小时前
蓝桥杯 Java B 组之设计 LRU 缓存
java·算法·蓝桥杯
万事可爱^7 小时前
HDBSCAN:密度自适应的层次聚类算法解析与实践
算法·机器学习·数据挖掘·聚类·hdbscan