参考:信号量
一、信号量
它是一个计数器。信号量用于实现进程间的互斥与同步,而不是用于存储进程间通信数据。
信号量是一个特殊的变量,程序对其访问都是原子操作,且只允许对它进行等待(即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的整数倍。