一.概述                                                   

System V信号量与System V消息队列不同。它不是用来在进程间传递数据。它主要是来同步进程的动作。

1.一个信号量是一个由内核维护的整数。其值被限制为大于或等于0。

2.可以在信号量上加上或减去一个数量。

3.当一个减操作把信号量减到小于0时,内核会阻塞调用进程。直到另一操作把信号恢复,阻塞才会解除。

4.常用的信号量是二进制信号量。即操作0和1来控制临界区。

二.函数接口                                           

1.创建或打开一个信号量

 #include <sys/sem.h>

 int semget(key_t key, int nsems, int semflg);

key和semflg跟消息队列一样,这里不再介绍,上面已给出消息队列的文章连接。

nsems:指定信号量数目,一般都是1。

2.控制信号量

 #include <sys/sem.h>

 int semctl(int semid, int semnum, int cmd, ...);

semid:semget()返回的信号量标识符。

semnum:信号量编号,如果是成组的信号量,就要用到它,否则是0。

cmd:控制信号量的命令。SETVAL初始化一个值,IPC_RMID删除信号量。

第四个参数是一个union semun结构。如果有的Linux版本头文件没有这个结构,需要自己定义:

 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) */
  };

3.改变信号量的值

 #include <sys/sem.h>

 int semop(int semid, struct sembuf *sops, size_t nsops);

sops:指向一个sembuf结构,该结构成员如下:

unsigned short sem_num:信号量编号,如果不是一组信号,一般都为0。
short sem_op:对信号量加减操作,如:-1,+1,1。
short sem_flg:通常设置为SEM_UNDO。如果进程终止时没有释放该信号量,内核会释放它。

三.简单例子                                            

我们封装一个简单的二进制信号量,写2个小程序,一个只打印基数,一个只打印偶数,通过二进制信号量来控制它们按顺序打印1-10。

1.封装二进制信号量

 /**
  * @file binary_sem.h
  */

 #ifndef _BINARY_SEM_H_
 #define _BINARY_SEM_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) */
 };

 /* 设置信号量 */
 int sem_set(int sem_id);
 /* 增大信号量 */
 int sem_up(int sem_id);
 /* 减小信号量 */
 int sem_down(int sem_id);
 /* 删除信号量 */
 int sem_delete(int sem_id);

 #endif
 /**
  * @file binary_sem.c
  */

 #include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
 #include <sys/sem.h>

 #include "binary_sem.h"

 static union semun sem_union;
 static struct sembuf sem_buf;

 /* 设置信号量 */
 int sem_set(int sem_id)
 {
     sem_union.val = ;
     , SETVAL, sem_union) == -)
         ;
     ;
 }

 /* 改变信号量为1 */
 int sem_up(int sem_id)
 {
     sem_buf.sem_num = ;
     sem_buf.sem_flg = SEM_UNDO;
     sem_buf.sem_op = ;
     ) == -)
         ;
     ;
 }

 /* 信号量减1 */
 int sem_down(int sem_id)
 {
     sem_buf.sem_num = ;
     sem_buf.sem_flg = SEM_UNDO;
     sem_buf.sem_op = -;
     ) == -)
         ;
     ;
 }

 /* 删除信号量 */
 int sem_delete(int sem_id)
 {
     , IPC_RMID, sem_union) == -)
         ;
     ;
 }

2.打印基数的程序

 /**
  * @file sem1.c
  */

 #include <stdio.h>
 #include <string.h>
 #include <stdlib.h>
 #include <sys/sem.h>
 #include <unistd.h>

 #include "binary_sem.h"

 #define SEM_KEY 7788

 void err_exit(const char *err_msg)
 {
     printf("error: %s\n", err_msg);
     exit();
 }

 int main(int argc, const char *argv[])
 {
     int sem_id, i;

     /* 创建信号 */
     ,  | IPC_CREAT)) == -)
         err_exit("semget()");

     /* 初始化信号为1 */
     )
         err_exit("sem_set()");

     ; i < ; i += )
     {
         /* 信号减一 */
         )
             err_exit("sem_down()");

         sleep();
         printf(], i);

         /* 信号加一 */
         )
             err_exit("sem_up()");

         sleep();
     }
     ;
 }

对信号量加减操作之间的代码就是临界区。程序第39行的sleep()是为了有足够的时间来允许第二个程序,46行的sleep()是尽量让其他程序来获取信号量。

3.只打印偶数的程序

 /**
  * @file sem2.c
  */

 #include <stdio.h>
 #include <string.h>
 #include <stdlib.h>
 #include <unistd.h>
 #include <sys/sem.h>

 #include "binary_sem.h"

 #define SEM_KEY 7788

 void err_exit(const char *err_msg)
 {
     printf("error: %s\n", err_msg);
     exit();
 }

 int main(int argc, const char *argv[])
 {
     int sem_id, i;

     /* 创建信号 */
     ,  | IPC_CREAT)) == -)
         err_exit("semget()");

     ; i <= ; i += )
     {
         /* 信号减一 */
         )
             err_exit("sem_down()");

         printf(], i);

         /* 信号加一 */
         )
             err_exit("sem_up()");

         sleep();
     }
     ;
 }

四.实验                                                  

1.为了测试方便,上面2个程序的key统一用一个值为7788的宏来指定。

2.两个程序执行的流程:sem1初始化一个为1的信号量,进入循环后对信号执行减一,并进入临界区。休眠3秒的时间我执行第二个程序sem2,sem2进入循环后也执行减一操作,但此时信号已被sem1减为0,此时内核会阻塞sem1的操作,因为再减就小于0了。回到sem1,3秒过后,打印基数,并把信号重置为1,离开临界区,休眠1秒。此时内核发现信号是1,可以执行减操作,内核会对sem2放行,sem2进入临界区打印偶数。刚刚休眠1秒的sem1再次进入临界区,发现信号量被sem2减到了0,此时被内核阻塞,直到sem2重置信号量。如此反复!!!

最后:其实System V信号量的语义和API都比较复杂,有机会再探讨比它简单的POSIX信号量。

System V IPC(2)-信号量的更多相关文章

  1. System V IPC 之信号量

    本文继<System V IPC 之共享内存>之后接着介绍 System V IPC 的信号量编程.在开始正式的内容前让我们先概要的了解一下 Linux 中信号量的分类. 信号量的分类 在 ...

  2. Linux 系统编程 学习:05-进程间通信2:System V IPC(2)

    Linux 系统编程 学习:05-进程间通信2:System V IPC(2) 背景 上一讲 进程间通信:System V IPC(1)中,我们介绍了System IPC中有关消息队列.共享内存的概念 ...

  3. 线程同步、信号量、system v IPC

    一.线程同步 条件变量 什么是条件变量? 线程A等待某个条件成立,条件成立,线程A才继续向下执行.线程B的执行使条件成立,条件成立以后唤醒线程A,以继续执行.这个条件就是条件变量. pthread_c ...

  4. 第3章 System V IPC

    3.1 概述 System V IPC 包含:System V消息队列.System V信号量.System V共享内存. 3.2 key_t 键和 ftok函数 这三种类型的System V IPC ...

  5. 《Unix网络编程》卷2 读书笔记 第3章- System V IPC

    1. 概述 三种类型的System V IPC:System V 消息队列.System V 信号量.System V 共享内存区 System V IPC在访问它们的函数和内核为它们维护的信息上共享 ...

  6. 从并发处理谈PHP进程间通信(二)System V IPC

    .container { margin-right: auto; margin-left: auto; padding-left: 15px; padding-right: 15px } .conta ...

  7. System V IPC 之共享内存

    IPC 是进程间通信(Interprocess Communication)的缩写,通常指允许用户态进程执行系列操作的一组机制: 通过信号量与其他进程进行同步 向其他进程发送消息或者从其他进程接收消息 ...

  8. System V IPC 之消息队列

    消息队列和共享内存.信号量一样,同属 System V IPC 通信机制.消息队列是一系列连续排列的消息,保存在内核中,通过消息队列的引用标识符来访问.使用消息队列的好处是对每个消息指定了特定消息类型 ...

  9. 四十九、进程间通信——System V IPC 之消息队列

    49.1 System V IPC 介绍 49.1.1 System V IPC 概述 UNIX 系统存在信号.管道和命名管道等基本进程间通讯机制 System V 引入了三种高级进程间通信机制 消息 ...

随机推荐

  1. SQL数据库: 错误2812 未能找到存储过程 sp_password

    SET QUOTED_IDENTIFIER ON  GO  SET ANSI_NULLS OFF  GO  create procedure sp_password      @old sysname ...

  2. SNMPv3/pygal制图/smtplib发邮件

    SNMPv3 SNMPv3在路由器端的配置 这个我都没配置过,还得现学现卖这个链接说的是SNMPv3的基本配置 这个链接说的是SNMPv3的view命令,用于管理员可以看到哪些层级的内容 1) 配置 ...

  3. Dewplayer 音乐播放器

    Dewplayer 是一款用于 Web 的轻量级 Flash 音乐播放器.提供有多种样式选择,支持播放列表,并可以通过 JavaScript 接口来控制播放器. 注意事项: 该播放器只支持 mp3 格 ...

  4. Quartz.NET开源作业调度框架系列(三):IJobExecutionContext 参数传递

    前面写了关于Quartz.NET开源作业调度框架的入门和Cron Trigger , 这次继续这个系列, 这次想讨论一下Quartz.NET中的Job如何通过执行上下文(Execution Conte ...

  5. 图解javascript this指向什么?

    JavaScript 是一种脚本语言,支持函数式编程.闭包.基于原型的继承等高级功能.JavaScript一开始看起来感觉会很容易入门,但是随着使用的深入,你会发现JavaScript其实很难掌握,有 ...

  6. JS中取整以及随机颜色问题

    前言:感觉自己已经好久好久没有写博客了,最近都是在写在线笔记比较多.现在来到新公司了,昨天刚刚完成一个项目所以今天有空研究研究一下前端方面的技术.下午在看一个游戏代码的时候,发现了几个别人留下的不错的 ...

  7. Dotfuscator混淆加密

    混淆加密 1. 需要安装Dotfuscator软件 2. 安装好后打开软件,找到编译好的DLL文件 3. 打开[setting]设置属性,如下图: 把 Disable String Encryptio ...

  8. 导入CSV格式的数据

    导入CSV格式的数据 (参见http://dev.mysql.com/doc/refman/5.6/en/load-data.html) 1.数据库表(st_pptn_r) CREATE TABLE ...

  9. c++引用总结

    一.变量的引用: 引用:就是某一变量(目标)的一个别名,对引用的操作与对变量直接操作完全一样. 引用的声明方法:类型标识符 &引用名=目标变量名:  例:char ch;        cha ...

  10. Android提交数据到JavaWeb服务器实现登录

    之前学习Android提交数据到php服务器没有成功,在看了两三个星期的视频之后,现在终于实现了与服务器的交互.虽然完成的不是PHP端的,但是在这个过程还是学到了不少东西的.现在我先来展示一下我的成果 ...