linux 进程同步中的信号量

本贴最后更新于 2354 天前,其中的信息可能已经天翻地覆

ps:这篇博文主要是来记录在学校中学习的信号量机制的,哈哈哈:joy:

信号量机概念是由荷兰科学家 Dijkstr 引入,值得一提的是,它提出的 Dijksrtr 算法解决了最短路径问题。

      信号量又称为信号灯,它是用来协调不同进程间的数据对象的,而最主要的应用是共享内存方式的进程间通信。本质上,信号量是一个计数器,它用来记录对某个资源(如共享内存)的存取状况,信号量是一个特殊的变量,并且只有两个操作可以改变其值:等待(wait)与信号(signal)。

 

因为在 Linux 与 UNIX 编程中,"wait"与"signal"已经具有特殊的意义了(暂不知这特殊意义是啥),所以原始概念:

    用于等待(wait)的 P(信号量变量) ;
    用于信号(signal)的 V(信号量变量) ;

这两字母来自等待(passeren:通过,如同临界区前的检测点)与信号(vrjgeven:指定或释放,如同释放临界区的控制权)的荷兰语。

 

P 操作 负责把当前进程由运行状态转换为阻塞状态,直到另外一个进程唤醒它。

操作为:申请一个空闲资源(把信号量减 1),若成功,则退出;若失败,则该进程被阻塞;

 

V 操作 负责把一个被阻塞的进程唤醒,它有一个参数表,存放着等待被唤醒的进程信息。

操作为:释放一个被占用的资源(把信号量加 1),如果发现有被阻塞的进程,则选择一个唤醒之。 

补充:查看共享信息的内存的命令是 ipcs [-m|-s|-q] (全部的话是 ipcs -a) ;查看共享信息的内存的命令是 ipcs [-m|-s|-q]。

示例代码:

//testsem.c 主程序,使用PV操作实现三个进程的互斥 #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> #include <errno.h> #define SEMPERM 0600 #define TRUE 1 #define FALSE 0 typedef union _semun { int val; struct semid_ds *buf; ushort *array; } semun; void handlesem(key_t skey); int initsem(key_t semkey); int p(int semid); int v(int semid); main() { key_t semkey=0x400; int i; for (i=0;i<3;i++) { if (fork()==0) //父进程负责产生3个子进程 handlesem(semkey); //子进程中才执行handlesem,做完后就exit。 } } void handlesem(key_t skey) { int semid; pid_t pid=getpid(); if ((semid=initsem(skey))<0) exit(1); printf("进程 %d 在临界资源区之前 \n",pid); p(semid); //进程进入临界资源区,信号量减少1 printf("进程 %d 在使用临界资源时,停止10s \n",pid); /*in real life do something interesting */ sleep(10); printf("进程 %d 退出临界区后 \n",pid); v(semid); //进程退出临界资源区,信号量加1 printf("进程 %d 完全退出\n",pid); exit(0); } int initsem(key_t semkey) { int status=0,semid; //信号量标识符semid if ((semid=semget(semkey,1,SEMPERM|IPC_CREAT|IPC_EXCL))==-1) { if (errno==EEXIST) //EEXIST:信号量集已经存在,无法创建 semid=semget(semkey,1,0); //创建一个信号量 } else { semun arg; arg.val=1; //信号量的初值 status=semctl(semid,0,SETVAL,arg); //设置信号量集中的一个单独的信号量的值。 } if (semid==-1||status==-1) { perror("initsem failed"); return(-1); } /*all ok*/ return(semid); } int p(int semid) { struct sembuf p_buf; p_buf.sem_num=0; p_buf.sem_op=-1; //信号量减1,注意这一行的1前面有个负号 p_buf.sem_flg=SEM_UNDO; //p_buf = {0,-1,SEM_UNDO}; if (semop(semid, &p_buf, 1)==-1) { perror("p(semid)failed"); exit(1); } return(0); } int v(int semid) { struct sembuf v_buf; v_buf.sem_num=0; v_buf.sem_op=1; //信号量加1 v_buf.sem_flg=SEM_UNDO; if (semop(semid, &v_buf, 1)==-1) { perror("v(semid)failed"); exit(1); } return(0); }

运行结果:
1png

2png

3png

相关说明
(一)系统调用函数 semget()
函数原型:int semget(key_t key,int nsems,int semflg);

功能描述: 创建一个新的信号量集,或者存取一个已经存在的信号量集。

当调用 semget 创建一个信号量时,他的相应的 semid_ds 结构被初始化。ipc_perm 中各个量被设置为相应
值:
        sem_nsems 被设置为 nsems 所示的值;    
        sem_otime 被设置为 0; 
        sem_ctime 被设置为当前时间

参数介绍:
         key:所创建或打开信号量集的键值,键值是 IPC_PRIVATE,该值通常为 0,创建一个仅能被进程进程给我的信号量, 键值不是 IPC_PRIVATE,我们可以指定键值,例如 1234;也可以一个 ftok()函数来取得一个唯一的键值。
         nsems:创建的信号量集中的信号量的个数,该参数只在创建信号量集时有效。
         semflg:调用函数的操作类型,也可用于设置信号量集的访问权限,两者通过 or 表示:

                有 IPC_CREAT,IPC_EXCL 两种:

IPC_CREAT 如果信号量不存在,则创建一个信号量,否则获取。

IPC_EXCL 只有信号量不存在的时候,新的信号量才建立,否则就产生错误。

返回值说明:
如果成功,则返回信号量集的 IPC 标识符,其作用与信息队列识符一样。
如果失败,则返回-1,errno 被设定成以下的某个值
EACCES:没有访问该信号量集的权限
EEXIST:信号量集已经存在,无法创建
EINVAL:参数 nsems 的值小于 0 或者大于该信号量集的限制;或者是该 key 关联的信号量集已存在,并且 nsems
大于该信号量集的信号量数
ENOENT:信号量集不存在,同时没有使用 IPC_CREAT
ENOMEM :没有足够的内存创建新的信号量集
ENOSPC:超出系统限制

每个信号量都有一些相关值:

      semval 信号量的值,一般是一个正整数,它只能通过信号量系统调用 semctl 函数设置,程序无法直接对它进行修改。

      sempid 最后一个对信号量进行操作的进程的 pid.

      semcnt 等待信号量的值大于其当前值的进程数。

      semzcnt 等待信号量的值归零的进程数。

 
(二)信号量的控制 semctl()
原型:int semctl(int semid,int semnum,int cmd,union semun ctl_arg); 
参数介绍: semid 为信号量集引用标志符,即 semget 的返回值。 

               semnum 第二个参数是信号量数目;

               cmd 表示调用该函数执行的操作,其取值和对应操作如下:

标准的 IPC 函数

(注意在头文件 <sys/sem.h> 中包含 semid_ds 结构的定义)

IPC_STAT 把状态信息放入 ctl_arg.stat 中

IPC_SET 用 ctl_arg.stat 中的值设置所有权/许可权

IPC_RMID 从系统中删除信号量集合

单信号量操作

(下面这些宏与 sem_num 指定的信号量合 semctl 返回值相关)

GETVAL 返回信号量的值(也就是 semval)

SETVAL 把信号量的值写入 ctl_arg.val 中

GETPID 返回 sempid 值

GETNCNT 返回 semncnt(参考上面内容)

GETZCNT 返回 semzcnt(参考上面内容)

全信号量操作

GETALL 把所有信号量的 semvals 值写入 ctl_arg.array

SETALL 用 ctl_arg.array 中的值设置所有信号量的 semvals

 

参数 arg 代表一个 union 的 semun 的实例。semun 是在 linux/sem.h 中定义的:

union semun {
int val; //执行 SETVAL 命令时使用
struct semid_ds *buf; //在 IPC_STAT/IPC_SET 命令中使用
unsigned short *array; //使用 GETALL/SETALL 命令时使用的指针
}

联合体中每个成员都有各自不同的类型,分别对应三种不同的 semctl 功能,如果 semval 是 SETVAL.则使用的将是 ctl_arg.val.

。     

功能:smctl 函数依据 command 参数会返回不同的值。它的一个重要用途是为信号量赋初值,因为进程无法直接对信号量的值进行修改。

(三)信号量操作 semop 函数
在 Linux 下,PV 操作通过调用 semop 函数来实现,也只有它能对 PV 进行操作

调用原型:int semop(int semid,struct sembuf*sops,unsign ednsops);

返回值:0,如果成功。-1,如果失败:errno=E2BIG(nsops 大于最大的 ops 数目)
EACCESS(权限不够)
EAGAIN(使用了 IPC_NOWAIT,但操作不能继续进行)
EFAULT(sops 指向的地址无效)
EIDRM(信号量集已经删除)
EINTR(当睡眠时接收到其他信号)
EINVAL(信号量集不存在,或者 semid 无效)
ENOMEM(使用了 SEM_UNDO,但无足够的内存创建所需的数据结构)
ERANGE(信号量值超出范围)

参数介绍:

第一个参数 semid 是信号量集合标识符,它可能是从前一次的 semget 调用中获得的。

第二个参数是一个 sembuf 结构的数组,每个 sembuf 结构体对应一个特定信号的操作,sembuf 结构在,<sys/sem.h> 中定义

struct sembuf{
usign short sem_num;/信号量索引/
short sem_op;/要执行的操作/
short sem_flg;/操作标志/
}

sem_num 存放集合中某一信号量的索引,如果集合中只包含一个元素,则 sem_num 的值只能为 0。


Sem_op 取得值为一个有符号整数,该整数实际给定了 semop 函数将完成的功能。包括三种情况:

      如果 sem_op 是负数,那么信号量将减去它的值,对应于 p()操作。这和信号量控制的资源有关。如果没有使用 IPC_NOWAIT,那么调用进程将进入睡眠状态,直到信号量控制的资源可以使用为止。

      如果 sem_op 是正数,则信号量加上它的值。对应于 v()操作。这也就是进程释放信号量控制的资源。

      最后,如果 sem_op 是 0,那么调用进程将调用 sleep(),直到信号量的值为 0。这在一个进程等待完全空闲的资源时使用。


sem_flag 是用来告诉系统当进程退出时自动还原操作,它维护着一个整型变量 semadj(信号灯的计数器),可设置为 IPC_NOWAIT 或 SEM_UNDO 两种状态。只有将 sem_flg 指定为 SEM_UNDO 标志后,semadj (所指定信号量针对调用进程的调整值)才会更新,即减去减去 sem_num 的值。 此外,如果此操作指定 SEM_UNDO,系统更新过程中会撤消此信号灯的计数(semadj)。此操作可以随时进行---它永远不会强制等待的过程。调用进程必须有改变信号量集的权限。

 

第三个参数是 sembuf 组成的数组中索引。参数 sops 指向由 sembuf 组成的数组,结构数组中的一员。

整理自_linux 进程同步之信号量:http://www.cnblogs.com/LZYY/p/3453582.html

  • Linux

    Linux 是一套免费使用和自由传播的类 Unix 操作系统,是一个基于 POSIX 和 Unix 的多用户、多任务、支持多线程和多 CPU 的操作系统。它能运行主要的 Unix 工具软件、应用程序和网络协议,并支持 32 位和 64 位硬件。Linux 继承了 Unix 以网络为核心的设计思想,是一个性能稳定的多用户网络操作系统。

    950 引用 • 943 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖
  • Swagger

    Swagger 是一款非常流行的 API 开发工具,它遵循 OpenAPI Specification(这是一种通用的、和编程语言无关的 API 描述规范)。Swagger 贯穿整个 API 生命周期,如 API 的设计、编写文档、测试和部署。

    26 引用 • 35 回帖 • 2 关注
  • 服务

    提供一个服务绝不仅仅是简单的把硬件和软件累加在一起,它包括了服务的可靠性、服务的标准化、以及对服务的监控、维护、技术支持等。

    41 引用 • 24 回帖 • 1 关注
  • 酷鸟浏览器

    安全 · 稳定 · 快速
    为跨境从业人员提供专业的跨境浏览器

    3 引用 • 59 回帖 • 46 关注
  • Word
    13 引用 • 40 回帖
  • frp

    frp 是一个可用于内网穿透的高性能的反向代理应用,支持 TCP、UDP、 HTTP 和 HTTPS 协议。

    20 引用 • 7 回帖
  • 书籍

    宋真宗赵恒曾经说过:“书中自有黄金屋,书中自有颜如玉。”

    76 引用 • 389 回帖
  • Angular

    AngularAngularJS 的新版本。

    26 引用 • 66 回帖 • 542 关注
  • SSL

    SSL(Secure Sockets Layer 安全套接层),及其继任者传输层安全(Transport Layer Security,TLS)是为网络通信提供安全及数据完整性的一种安全协议。TLS 与 SSL 在传输层对网络连接进行加密。

    70 引用 • 193 回帖 • 412 关注
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 567 关注
  • SQLite

    SQLite 是一个进程内的库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是全世界使用最为广泛的数据库引擎。

    5 引用 • 7 回帖 • 2 关注
  • SpaceVim

    SpaceVim 是一个社区驱动的模块化 vim/neovim 配置集合,以模块的方式组织管理插件以
    及相关配置,为不同的语言开发量身定制了相关的开发模块,该模块提供代码自动补全,
    语法检查、格式化、调试、REPL 等特性。用户仅需载入相关语言的模块即可得到一个开箱
    即用的 Vim-IDE。

    3 引用 • 31 回帖 • 117 关注
  • ZeroNet

    ZeroNet 是一个基于比特币加密技术和 BT 网络技术的去中心化的、开放开源的网络和交流系统。

    1 引用 • 21 回帖 • 639 关注
  • Rust

    Rust 是一门赋予每个人构建可靠且高效软件能力的语言。Rust 由 Mozilla 开发,最早发布于 2014 年 9 月。

    58 引用 • 22 回帖 • 1 关注
  • CodeMirror
    1 引用 • 2 回帖 • 154 关注
  • iOS

    iOS 是由苹果公司开发的移动操作系统,最早于 2007 年 1 月 9 日的 Macworld 大会上公布这个系统,最初是设计给 iPhone 使用的,后来陆续套用到 iPod touch、iPad 以及 Apple TV 等产品上。iOS 与苹果的 Mac OS X 操作系统一样,属于类 Unix 的商业操作系统。

    87 引用 • 139 回帖 • 1 关注
  • Vue.js

    Vue.js(读音 /vju ː/,类似于 view)是一个构建数据驱动的 Web 界面库。Vue.js 的目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件。

    267 引用 • 666 回帖 • 1 关注
  • Kubernetes

    Kubernetes 是 Google 开源的一个容器编排引擎,它支持自动化部署、大规模可伸缩、应用容器化管理。

    116 引用 • 54 回帖 • 5 关注
  • WiFiDog

    WiFiDog 是一套开源的无线热点认证管理工具,主要功能包括:位置相关的内容递送;用户认证和授权;集中式网络监控。

    1 引用 • 7 回帖 • 606 关注
  • 阿里巴巴

    阿里巴巴网络技术有限公司(简称:阿里巴巴集团)是以曾担任英语教师的马云为首的 18 人,于 1999 年在中国杭州创立,他们相信互联网能够创造公平的竞争环境,让小企业通过创新与科技扩展业务,并在参与国内或全球市场竞争时处于更有利的位置。

    43 引用 • 221 回帖 • 75 关注
  • CentOS

    CentOS(Community Enterprise Operating System)是 Linux 发行版之一,它是来自于 Red Hat Enterprise Linux 依照开放源代码规定释出的源代码所编译而成。由于出自同样的源代码,因此有些要求高度稳定的服务器以 CentOS 替代商业版的 Red Hat Enterprise Linux 使用。两者的不同在于 CentOS 并不包含封闭源代码软件。

    239 引用 • 224 回帖 • 1 关注
  • 京东

    京东是中国最大的自营式电商企业,2015 年第一季度在中国自营式 B2C 电商市场的占有率为 56.3%。2014 年 5 月,京东在美国纳斯达克证券交易所正式挂牌上市(股票代码:JD),是中国第一个成功赴美上市的大型综合型电商平台,与腾讯、百度等中国互联网巨头共同跻身全球前十大互联网公司排行榜。

    14 引用 • 102 回帖 • 319 关注
  • Caddy

    Caddy 是一款默认自动启用 HTTPS 的 HTTP/2 Web 服务器。

    12 引用 • 54 回帖 • 168 关注
  • 七牛云

    七牛云是国内领先的企业级公有云服务商,致力于打造以数据为核心的场景化 PaaS 服务。围绕富媒体场景,七牛先后推出了对象存储,融合 CDN 加速,数据通用处理,内容反垃圾服务,以及直播云服务等。

    28 引用 • 226 回帖 • 138 关注
  • 微服务

    微服务架构是一种架构模式,它提倡将单一应用划分成一组小的服务。服务之间互相协调,互相配合,为用户提供最终价值。每个服务运行在独立的进程中。服务于服务之间才用轻量级的通信机制互相沟通。每个服务都围绕着具体业务构建,能够被独立的部署。

    96 引用 • 155 回帖 • 1 关注
  • 链滴

    链滴是一个记录生活的地方。

    记录生活,连接点滴

    171 引用 • 3842 回帖
  • JRebel

    JRebel 是一款 Java 虚拟机插件,它使得 Java 程序员能在不进行重部署的情况下,即时看到代码的改变对一个应用程序带来的影响。

    26 引用 • 78 回帖 • 676 关注