linux 进程同步中的信号量

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

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 以网络为核心的设计思想,是一个性能稳定的多用户网络操作系统。

    918 引用 • 931 回帖 • 2 关注

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • Hibernate

    Hibernate 是一个开放源代码的对象关系映射框架,它对 JDBC 进行了非常轻量级的对象封装,使得 Java 程序员可以随心所欲的使用对象编程思维来操纵数据库。

    39 引用 • 103 回帖 • 687 关注
  • LaTeX

    LaTeX(音译“拉泰赫”)是一种基于 ΤΕΧ 的排版系统,由美国计算机学家莱斯利·兰伯特(Leslie Lamport)在 20 世纪 80 年代初期开发,利用这种格式,即使使用者没有排版和程序设计的知识也可以充分发挥由 TeX 所提供的强大功能,能在几天,甚至几小时内生成很多具有书籍质量的印刷品。对于生成复杂表格和数学公式,这一点表现得尤为突出。因此它非常适用于生成高印刷质量的科技和数学类文档。

    9 引用 • 32 回帖 • 155 关注
  • Electron

    Electron 基于 Chromium 和 Node.js,让你可以使用 HTML、CSS 和 JavaScript 构建应用。它是一个由 GitHub 及众多贡献者组成的活跃社区共同维护的开源项目,兼容 Mac、Windows 和 Linux,它构建的应用可在这三个操作系统上面运行。

    15 引用 • 136 回帖 • 4 关注
  • WebClipper

    Web Clipper 是一款浏览器剪藏扩展,它可以帮助你把网页内容剪藏到本地。

    3 引用 • 9 回帖
  • sts
    2 引用 • 2 回帖 • 154 关注
  • 机器学习

    机器学习(Machine Learning)是一门多领域交叉学科,涉及概率论、统计学、逼近论、凸分析、算法复杂度理论等多门学科。专门研究计算机怎样模拟或实现人类的学习行为,以获取新的知识或技能,重新组织已有的知识结构使之不断改善自身的性能。

    76 引用 • 37 回帖
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖
  • 星云链

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

    3 引用 • 16 回帖 • 1 关注
  • Maven

    Maven 是基于项目对象模型(POM)、通过一小段描述信息来管理项目的构建、报告和文档的软件项目管理工具。

    186 引用 • 318 回帖 • 339 关注
  • Vditor

    Vditor 是一款浏览器端的 Markdown 编辑器,支持所见即所得、即时渲染(类似 Typora)和分屏预览模式。它使用 TypeScript 实现,支持原生 JavaScript、Vue、React 和 Angular。

    320 引用 • 1679 回帖 • 1 关注
  • Solidity

    Solidity 是一种智能合约高级语言,运行在 [以太坊] 虚拟机(EVM)之上。它的语法接近于 JavaScript,是一种面向对象的语言。

    3 引用 • 18 回帖 • 355 关注
  • Firefox

    Mozilla Firefox 中文俗称“火狐”(正式缩写为 Fx 或 fx,非正式缩写为 FF),是一个开源的网页浏览器,使用 Gecko 排版引擎,支持多种操作系统,如 Windows、OSX 及 Linux 等。

    7 引用 • 30 回帖 • 451 关注
  • 小说

    小说是以刻画人物形象为中心,通过完整的故事情节和环境描写来反映社会生活的文学体裁。

    28 引用 • 108 回帖 • 1 关注
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    129 引用 • 793 回帖
  • OnlyOffice
    4 引用 • 19 关注
  • WiFiDog

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

    1 引用 • 7 回帖 • 545 关注
  • Spring

    Spring 是一个开源框架,是于 2003 年兴起的一个轻量级的 Java 开发框架,由 Rod Johnson 在其著作《Expert One-On-One J2EE Development and Design》中阐述的部分理念和原型衍生而来。它是为了解决企业应用开发的复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 JavaEE 应用程序开发提供集成的框架。

    941 引用 • 1458 回帖 • 135 关注
  • 互联网

    互联网(Internet),又称网际网络,或音译因特网、英特网。互联网始于 1969 年美国的阿帕网,是网络与网络之间所串连成的庞大网络,这些网络以一组通用的协议相连,形成逻辑上的单一巨大国际网络。

    96 引用 • 330 回帖
  • NGINX

    NGINX 是一个高性能的 HTTP 和反向代理服务器,也是一个 IMAP/POP3/SMTP 代理服务器。 NGINX 是由 Igor Sysoev 为俄罗斯访问量第二的 Rambler.ru 站点开发的,第一个公开版本 0.1.0 发布于 2004 年 10 月 4 日。

    311 引用 • 546 回帖 • 34 关注
  • 游戏

    沉迷游戏伤身,强撸灰飞烟灭。

    169 引用 • 800 回帖 • 1 关注
  • 外包

    有空闲时间是接外包好呢还是学习好呢?

    26 引用 • 232 回帖 • 8 关注
  • 域名

    域名(Domain Name),简称域名、网域,是由一串用点分隔的名字组成的 Internet 上某一台计算机或计算机组的名称,用于在数据传输时标识计算机的电子方位(有时也指地理位置)。

    43 引用 • 208 回帖 • 1 关注
  • Openfire

    Openfire 是开源的、基于可拓展通讯和表示协议 (XMPP)、采用 Java 编程语言开发的实时协作服务器。Openfire 的效率很高,单台服务器可支持上万并发用户。

    6 引用 • 7 回帖 • 92 关注
  • abitmean

    有点意思就行了

    24 关注
  • PostgreSQL

    PostgreSQL 是一款功能强大的企业级数据库系统,在 BSD 开源许可证下发布。

    22 引用 • 22 回帖 • 2 关注
  • 链书

    链书(Chainbook)是 B3log 开源社区提供的区块链纸质书交易平台,通过 B3T 实现共享激励与价值链。可将你的闲置书籍上架到链书,我们共同构建这个全新的交易平台,让闲置书籍继续发挥它的价值。

    链书社

    链书目前已经下线,也许以后还有计划重制上线。

    14 引用 • 257 回帖 • 1 关注
  • flomo

    flomo 是新一代 「卡片笔记」 ,专注在碎片化时代,促进你的记录,帮你积累更多知识资产。

    4 引用 • 88 回帖 • 5 关注