Linux中的System V信号量
在进程同步,并发运行时,保证按序地访问共享资源是十分重要的。因此引入了临界区的概念,一次只能有一个线程进入临界区完成他的指令。而信号量(semaphore)的作用,类似于一个交通信号灯,它负责进程协作,因此信号量又称为信号灯。
在Linux系统中,它提供两种信号量:
内核信号量,由内核控制路径使用
用户态进程使用的信号量,这种信号量有两种接口,
POSIX信号量和SYSTEM V信号量。信号量的本质是一个计数器。一个较为常见的用法,是为每个资源都会分配一个信号量。记信号量为S,除了初始化之外,有两个标准原子操作:
wait()和signal()。
System V信号量接口
semget
创建一个新信号量或取得一个已有信号量
int semget(key_t key, int num_sems, int sem_flags);
key是一个整数值(唯一非零),可以理解成是信号量的标识符。num_sems指定了需要的信号量数目,通常为1。sem_flags是一组标志,当创建一个新的信号量时,设定权限与值IPC_CREAT做按位或操作。设置了IPC_CREAT标志后,即使给出的键是一个已有信号量的键,也不会产生错误。而IPC_CREAT | IPC_EXCL则可以创建一个新的,唯一的信号量,如果信号量已存在,返回一个错误。函数成功返回一个相应信号标识符(非零),失败返回
-1。semctl
直接控制信号量信息
int semctl(int sem_id, int sem_num, int command, ...);
第二个参数是操作信号在信号集中的编号,第一个信号的编号是
0。第三个参数
command通常是下面两个值中的其中一个:SETVAL:用来把信号量初始化为一个已知的值。IPC_RMID:用于删除一个已经无需继续使用的信号量标识符。如果有第四个参数,它通常是一个
union semum结构,定义如下:union semun
{
int val;
struct semid_ds *buf;
unsigned short *arry;
};
semop
改变信号量的值
int semop(int sem_id, struct sembuf *sem_opa, size_t num_sem_ops);
sem_id是由semget返回的信号量标识符,sembuf结构的定义如下:struct sembuf{
short sem_num;//除非使用一组信号量,否则为0
short sem_op;//信号量在一次操作中需要改变的数据,-1即P(等待)操作,+1即V(发送信号)操作。
short sem_flg;//通常为SEM_UNDO,使操作系统跟踪信号,并在进程没有释放该信号量而终止时,操作系统释放信号量
};
进程同步实例
无信号量实例
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
pid_t pid;
pid = fork();
srand(pid);
if(pid > 0) // parent process
{
char a = 'A'; // char to print
for(int i = 0; i < 10; ++i)
{
printf("%c", a);
fflush(stdout); // flush stdout buffer
sleep(1);
printf("%c", a);
fflush(stdout);
sleep(1);
}
}
else // child process
{
char b = 'B';
for(int i = 0; i < 10; ++i)
{
printf("%c", b);
fflush(stdout);
sleep(1);
printf("%c", b);
fflush(stdout);
sleep(1);
}
}
printf("\n%d - finished\n", getpid());
sleep(3);
return 0;
}
运行结果

有信号量实例
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#define SEMKEY 0x00002222 // set a key for semaphore
union semun // union for semaphore
{
int val;
struct semid_ds *buf;
unsigned short *array;
};
struct sembuf p = { 0, -1, SEM_UNDO};
struct sembuf v = { 0, +1, SEM_UNDO};
int main()
{
int sem_id = semget(SEMKEY, 1, 0666 | IPC_CREAT); // get semaphore
union semun sem_union;
sem_union.val = 1;
if(semctl(sem_id, 0, SETVAL, sem_union) < 0)
{
perror("semctl error");
return -1;
}
int pid;
pid = fork();
srand(pid);
if(pid > 0) // parent process
{
char a = 'A'; // char to print
for(int i = 0; i < 10; ++i)
{
if(semop(sem_id, &p, 1) < 0) // P operation
{
perror("semop p error");
return -1;
}
printf("%c", a);
fflush(stdout); // flush stdout buffer
sleep(1);
printf("%c", a);
fflush(stdout);
if(semop(sem_id, &v, 1) < 0) // V operation
{
perror("semop v error");
return -1;
}
sleep(1);
}
}
else // child process
{
char b = 'B'; // char to print
for(int i = 0; i < 10; ++i)
{
if(semop(sem_id, &p, 1) < 0) // P operation
{
perror("semop p error");
return -1;
}
printf("%c", b);
fflush(stdout); // flush stdout buffer
sleep(1);
printf("%c", b);
fflush(stdout);
if(semop(sem_id, &v, 1) < 0) // V operation
{
perror("semop v error");
return -1;
}
sleep(1);
}
}
printf("\n%d - finished\n", getpid());
sleep(3);
if (pid > 0)
{
system("ipcrm -S 0x00002222");
}
return 0;
}
运行结果

因为设定信号量的关系,一个线程在临界区内一定会执行两次print()操作,所以A或B一定成对出现。
Linux中的System V信号量的更多相关文章
- linux网络编程之system v信号量(一)
今天起,学习信号量相关的知识,下面开始: 关于信号量,在前面已经介绍过了,这里回顾一下: 通过上面的描述,很容易就能想到信号量的一上数据结构: 下面再来回顾一下P.V原语: 所谓的原语就是指这段代码是 ...
- Linux IPC实践(11) --System V信号量(1)
信号量API #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> int semget ...
- linux网络编程之system v信号量(二)
今天迎来元旦假期的最后一天了,过得好快~昨天跟小伙伴们在军都滑雪陪儿爽,虽说上了两回中级道都摔得异常的惨烈,但是在初级道上学习"s"转弯还是有一些小心得,可以在要往高手迈进的前提, ...
- system V信号量和Posix信号量
一.函数上的区别 信号量有两种实现:传统的System V信号量和新的POSIX信号量.它们所提供的函数很容易被区分:对于所有System V信号量函数,在它们的名字里面没有下划线.例如,应该是sem ...
- System V信号量
信号量对比 二值信号量:其值要么0要么1,比如互斥锁就是这种类型 计数信号量:其值为0或某个正整数,比如POSIX 信号量 计数信号量:一个或多个信号量构成一个集合,每个都是计数信号量,比如Syste ...
- UNIX环境高级编程——system V信号量
1. 信号量(semaphore)主要用于保护临界资源.进程可以根据它判断是否能访问某些共享资源.信号量除了用于访问控制外,还可用于进程同步,也就是进程间通信.2. 信号量分类:a. 二值信号量: 信 ...
- System V 信号量使用相关函数
System V 信号量 在提到Posix 信号量时,指的是二值信号量或计数信号量,而System V信号量指的是入了计数信号量集 二值信号量:其值为0或1,类似于互斥锁,资源被锁住时为0,资源可用为 ...
- 第11章 System V 信号量
11.1 概述 信号量按功能分:二值信号量.计数信号量.信号量集:其中二值信号量和计数信号量指的是Posix信号量,信号量集指的是System V信号量.
- Linux IPC System V 信号量
模型 #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> ftok() //获取key ...
随机推荐
- Java数据结构(十二)—— 霍夫曼树及霍夫曼编码
霍夫曼树 基本介绍和创建 基本介绍 又称哈夫曼树,赫夫曼树 给定n个权值作为n个叶子节点,构造一棵二叉树,若该树的带权路径长度(wpl)达到最小,称为最优二叉树 霍夫曼树是带权路径长度最短的树,权值较 ...
- 三万字无坑搭建基于Docker+K8S+GitLab/SVN+Jenkins+Harbor持续集成交付环境
写在前面 最近在 K8S 1.18.2 版本的集群上搭建DevOps环境,期间遇到了各种坑.目前,搭建环境的过程中出现的各种坑均已被填平,特此记录,并分享给大家! 文章和搭建环境所需要的yml文件已收 ...
- Spring Boot 2.4.0 发布,配置文件重大调整,不要乱升级!!
前段时间 Spring Boot 2.4.0 发布了,栈长作了一个新特性全盘解读,其中介绍了一个很重要的变革,那就是配置文件. 配置文件可是每个框架的核心,不得不搞清楚,所以,这篇栈长就带大家深入实战 ...
- Markdown实用教程
Markdown 是用来编写结构化文档的一种纯文本格式,它使我们在双手不离开键盘的情况下,可以对文本进行一定程度的格式排版.markdown语法是通用的,很多写作平台都是支持markdown的,比如简 ...
- 第二十三章、 Model/View便利类表格部件QTableWidget详解
老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 一.引言 表格部件为应用程序提供标准的表格显示工具,在表格内可以管理基于行和列的数据项,表格中的最大 ...
- MySQL Docker容器实例创建并进入MySQL命令行
首先需要明白的一点是: docker镜像是一个模版,docker容器是一个实例,它可以被启动与关闭. 我们需要先有MySQL的docker镜像,使用命令: docker pull mysql 拉取最新 ...
- Alpha冲刺阶段Day4
[Alpha冲刺阶段]Scrum Meeting Daily4 1.会议简述 会议开展时间 2020/5/25 7:30-7:50 PM 会议基本内容摘要 讨论了各自任务完成情况以及明日计划 参与 ...
- Panda Global获悉,美国承诺4年内明确区块链数字资产监管方式!
近日,美国商品期货交易委员会(CFTC)宣布,在4年内将会全面把加密货币监管列为优先事项.Panda Global从7月8日公布的新战略中获悉,此次CFTC公布了自己接下来的新框架,并且在框架中承诺: ...
- JSON类型解析
JSON类型解析 一.序列化和反序列化 程序中的对象,如python中的字典.列表.函数.类等,都是存在内存中,不方便传递或存储,所以需要将内存中的对象转化为文本或者文件格式,来满足传输和持久化(存储 ...
- Windows单机安装hadoop
版本信息 Hadoop 3.2.0 java version "1.8.0_201" Windows 7专业版,64位 安装过程 jdk安装 下载jdk,解压到目录,D:\Java ...