进程间通信(InterProcess Communication,IPC):信号量、共享内存

本文详细介绍了Linux中的信号量和共享内存概念,包括它们的原理、操作方法(如semget、semop、shmat等)以及各自的优缺点,特别强调了如何使用信号量进行进程间同步和数据保护。
摘要由CSDN通过智能技术生成

参考:信号量

一、信号量

它是一个计数器。信号量用于实现进程间的互斥与同步,而不是用于存储进程间通信数据。

信号量是一个特殊的变量,程序对其访问都是原子操作,且只允许对它进行等待(即P(信号变量))和发送(即V(信号变量))信息操作。

特点:

  • 信号量用于进程间同步,若要在进程间传递数据需要结合共享内存。
  • 信号量基于操作系统的 PV 操作,程序对信号量的操作都是原子操作。
  • 每次对信号量的 PV 操作不仅限于对信号量值加 1 或减 1,而且可以加减任意正整数。
  • 支持信号量组。

原型
最简单的信号量是只能取 0 和 1 的变量,这也是信号量最常见的一种形式,叫做二值信号量(Binary Semaphore)。而可以取多个正整数的信号量被称为通用信号量。

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

#include <sys/sem.h>
int semget(key_t key, int num_sems, int sem_flags);// 创建或获取一个信号量组:若成功返回信号量集ID,失败返回-1
	//key:可以理解为文件目录索引值
	//num_sems: 指定需要的信号量数目,它的值几乎总是1。
	//sem_flags:一组标志,当想要当信号量不存在时创建一个新的信号量  IPC_CREAT|0666
int semop(int semid, struct sembuf semoparray[], size_t numops);  // 对信号量组进行操作,改变信号量的值,进行PV操作的函数:成功返回0,失败返回-1
	//sem_id:  是由semget返回的信号量标识符
	//sem_opa:信号量结构体struct sembuf指针,该指针改变后的信号量
 	//num_sem_ops: struct sembuf变量成员数量
int semctl(int semid, int sem_num, int cmd, ...);// 控制信号量的相关信息,初始化和删除信号量
	//sem_id:信号量标示符
	//sem_num:信号量集中有多个信号量,表示第几个信号量 0-*

信号量操作(semop)

  • P操作:sem_op= -1 <0,使得共享临界资源上锁,其它进程不得访问
  • V操作:sem_op= 1 >0,使得共享临界资源解锁,其它进程可以访问

控制信号量信息(semctl):senctl 函数的cmd 参数有如下选择:

  • SETVAL:用来把信号量初始化为一个已知的值。这个值通过union semun_t 中的val成员设置,其作用是在信号量第一次使用前对它进行设置。
  • GETVAL:获取信号集中的信号量的计数值。
  • IPC_RMID:用于删除一个已经无需继续使用的信号量标识符。
  • IPC_STAT:把semid_ds结构中的数据设置为信号集的当前关联值。
  • GETALL:用于读取信号量集中的所有信号量的值。
  • GETNCNT:返回正在等待资源的进程数目。
  • GETPID:返回最后一个执行semop操作的进程的PID。
  • GETZCNT:返回正在等待完全空闲的资源的进程数目。
  • SETALL:设置信号量集中的所有的信号量的值。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>       
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <unistd.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) */
};
void plock(int semid){   //P操作,初始值-1,之后<0,阻塞。钥匙减一
	struct sembuf sops;
	sops.sem_num=0;
	sops.sem_op=-1;
	sops.sem_flg=SEM_UNDO;
	semop(semid,&sops,1);	
}
void vfree(int semid){// V 操作加一,初始值>0,解放资源。钥匙加一
	struct sembuf sops;
	sops.sem_num=0;
	sops.sem_op=1;
	sops.sem_flg=SEM_UNDO;
	semop(semid,&sops,1);	
}
int main()
{
	int semid;
	int nExisted = 0;
	key_t key= 0xEB01;
	if((semid=semget(key,1,IPC_CREAT|IPC_EXCL|0666))==-1)//创建信号量
	{
		if(EEXIST == errno)
		{
			nExisted = 1; 
			if((semid=semget(key,1,IPC_CREAT|IPC_EXCL|0666))==-1)//信号量已存在,获取信号量
			{
				printf("semget existed error!!!");
				return -1;
			}
		}
		else
		{
			printf("semget excluded error!!!");
			return -1;
		}
	}
	
	if(0 == nExisted)
	{
		union semun initsem;
		initsem.val=0;
		semctl(semid,0,SETVAL,&initsem); //赋初始值==0.也可为1 ,看需求。相当于钥匙
 	}
 	
	plock(semid);
	printf("This is father\n");
	vfree(semid);
		
    semctl(semid,0,IPC_REID)//删除信号量,
	return 0;
}

二、共享内存

转自:SystemV 共享内存

共享内存——>约定好使用同一个唯一key,来进行通信的

原型

#include <sys/shm.h>

int shmget(key_t key, size_t size, int flag);// 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
	//key: 唯一标识码,类似密码的概念,使用此,不同进程可访问一块共享内存
	//size:是该共享存储段的长度,一般建议是4KB的整数倍
void *shmat(int shm_id, const void *addr, int flag);// 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1

int shmdt(void *addr);// 断开与共享内存的连接:成功返回0,失败返回-1

int shmctl(int shm_id, int cmd, struct shmid_ds *buf);// 控制共享内存的相关信息:成功返回0,失败返回-1

共享内存的生命周期

共享内存不属于任何进程,不归进程管理,除非调用释放函数,否则要不要释放是OS决定的,因此共享内存的生命周期是随内核的!!

释放的方式有三种:

  • 关机
  • 调用释放共享内存的函数 shmctl
  • 命令行释放 ipcrm -m 共享内存的id

共享内存的优缺点

1. 优点:
  • 共享内存是所有进程间通信中速度最快的。(无需缓冲区,能大大减少通信数据的拷贝次数)
2. 缺点:
  • 如果服务端读取速度较快,用户端发送数据较慢,就会产生同一段消息被服务端读取多遍。共享内存是不进行同步和互斥的,没有对数据进行任何保护。若想实现同步和互斥,需要结合信号量一起使用。

共享内存大小的建议

因为系统分配共享内存是以4KB为基本单位,一般建议申请共享内存的大小为4KB的整数倍。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值