Linux--信号量

2023-09-17 17:04:25

一、信号量
信号量(semaphore)与已经介绍过的IPC结构不同,他是一个计数器。
用于实现进程间的互斥与同步,而不是用于存储进程间的通信数据。
可以与共享内存配合使用。
临界资源: 多道程序系统种存在许多进程,他们共享各种资源,然而有很多资源一次智能供一个进程使用。一次仅允许一个进程使用的资源称为临界资源。许多物理设备都属于临界资源,如输入机、打印机、磁带机等。
信号量集: 相当于信号量的集合。
P V 操作 P拿取,V放回
1、特点:
①、信号量用于进程间同步,若要在进程间传递数据需要结合共享内存。
②、信号量基于操作系统的PV操作,程序对信号量的操作都是原子操作。
③、每次对信号量的PV操作不仅限于对信号量加 1 或减 1 ,而且可以加减任意正整数。
④、支持信号量组

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

#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、例:

//信号量

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

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

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

//修改程序,使信号量控制让子进程先运行:封装两个函数 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;
}
更多推荐

OpenCV自学笔记二十:图像分割和提取

1、用分水岭算法实现图像分割与提取分水岭算法是一种经典的图像分割算法,用于将图像中的前景和背景进行分离。它基于图像中的灰度值和梯度信息来确定边界,并通过填充区域将图像分割成多个连通的区域。以下是分水岭算法的基本原理:1.预处理:首先对输入图像进行预处理操作,例如灰度化、平滑滤波和边缘检测等,以便更好地捕捉图像的特征。2

新工具 !一键无限重置 Jetbrain 2023 最新版系列

今天逛github,看到了一个新的Jetbrains系列软件的无限30天试用的方法,体验了下,感觉还不错,使用方法很简单。我看介绍软件还处于测试阶段,大家感兴趣的可以试试看。演示软件RubyMine2023.1JetbrainKiller0.5.0使用方法软件名称叫JetbrainKiller,使用方法就是打开软件,一

基于 Socket 网络编程

基于Socket网络编程前言一、基于Socket的网络通信传输(传输层)二、UDP的数据报套接字编程1、UDP套接字编程API2、使用UDPSocket实现简单通信三、TCP流套接字编程1、TCP流套接字编程API2、使用TCPSocket实现简单通信3、使用Tcp协议进行网络传输的“五大要点”前言我们再进行网络编程时

VFP保存大文件到MSSQL,最大2G,超过得上手段

目前社群在聊这个大文件读取的问题,赵总说要把image字段换成BASE64来读取,我就一脸蒙。为什么要转BASE64,体积暴涨三分之一,明显直取更快,200MB文件对单个文件来说,不算大。赵总还写了啥直读程序,于是就来验证一下情况拿出猫框,简单的操作一下建立MSSQL数据库表1生成猫框DAL类DefineClassDa

思腾云计算

为推动AI行业的国产化布局,迎合国产化服务器的市场需求,思腾合力推出华思系列服务器。1.前置24盘12GSASEXP硬盘背板,可以插24个3.5/2.5寸SAS/SATA硬盘;2.后置12盘12GSASEXP硬盘背板,可插12个3.5/2.5寸SAS/SATA硬盘,外加4口U.2硬盘背板1套;3.内置2个M.2SATA

线性代数的本质(十一)——复数矩阵

文章目录复数矩阵附录极大线性无关组向量叉积复数矩阵矩阵AAA的元素aij∈Ca_{ij}\in\Complexaij​∈C,称为复矩阵。现将实数矩阵的一些概念推广到复数矩阵,相应的一些性质在复数矩阵同样适用。定义:设复矩阵A=(aij)m×nA=(a_{ij})_{m\timesn}A=(aij​)m×n​矩阵Aˉ=(

SkyWalking快速上手(五)——存放在内存、数据持久化

文章目录存放在内存一、概述二、数据存放方式1.指标数据2.跟踪数据三、优势和注意事项四、总结数据持久化一、指标数据的持久化二、跟踪数据的持久化三、注意事项四、总结存放在内存一、概述SkyWalking是一个开源的分布式系统追踪和性能监控工具,用于帮助开发人员和运维人员监控和分析分布式系统的性能问题。在SkyWalkin

SkyWalking入门之Agent原理初步分析

一、简介当前稍微上点体量的互联网公司已经逐渐采用微服务的开发模式,将之前早期的单体架构系统拆分为很多的子系统,子系统封装为微服务,彼此间通过HTTP协议RESETAPI的方式进行相互调用或者gRPC协议进行数据协作。早期微服务只有几个的情况下,我们遇到问题可以直接简单、快速地通过采集日志进行分析,是A服务存在问题还是B

ReadWriteLock(读写锁)和阻塞队列BlockingQueue与同步队列SynchronousQueue

1.ReadWriteLockpackagecom.kuang.rw;importjava.util.HashMap;importjava.util.Map;importjava.util.concurrent.locks.ReadWriteLock;importjava.util.concurrent.locks.R

传导和辐射EMI有什么区别?

当我们设计原型或使用开发板时,通常可以忽略电磁干扰。但EMI在现实生活中的电子设备和系统中是一个重要的主题,工程师有责任确保电路能够在预期的EMI水平下正常运行,并且不会产生过多的EMI。我倾向于将EMI与无线干扰联系起来,考虑到名称,这并不令人惊讶:它被称为电磁干扰,我们自然将其与电磁辐射联系起来。但正如您从本文标题

实在智能携手40+央企,探索财务大模型及数智化实践与应用

“这次培训给我一个最大的感触就是,过去以为AI智能化、大模型技术是很高深的事情。但现在,我们通过RPA等数字化工具,自主根据自己的工作岗位,完成业务自动化流程的开发和设计。AI技术没有想象中的那么难入门。”这是一位参加了“财务大模型及AI+RPA数智化实践与应用”专题研修班的学员,培训后有感而发的心得。探索财务数智化落

热文推荐