1.特点:  信号灯集,是控制访问临界资源

信号灯(semaphore),也叫信号量。它是不同进程间或一个给定进程内部不同线程间同步的机制System V的信号灯是一个或者多个信号灯的一个集合(允许对集合中的多个计数信号灯进行同时操作)。其中的每一个都是单独的计数信号灯。而Posix信号灯指的是单个计数信号灯

2.信号灯种类:
  posix有名信号灯
  posix基于内存的信号灯(无名信号灯)(线程)
  System V信号灯(IPC对象)(信号灯的值就是代表资源的数量)

3.创建步骤:
  1)产生key值
  2)创建信号灯集
  3)初始化信号灯(对信号灯集中的每个信号灯进行初始化)
  4)信号灯的P-V操作
  5)删除信号灯集

4.相关函数:
  1)int semget(key_t key, int nsems, int semflg);
    功能:创建/打开信号灯
    参数:key:ftok产生的key值(和信号灯关联的key值)
         nsems:信号灯集中包含的信号灯数目
         semflg:信号灯集的访问权限,通常为IPC_CREAT |0666
    返回值:成功:信号灯集ID ; 失败:-1

  2)int semop ( int semid, struct sembuf *opsptr, size_t nops);
    功能:对信号灯集合中的信号量进行P - V操作
    参数:semid:信号灯集ID
         struct sembuf {
        short sem_num; // 要操作的信号灯的编号
        short sem_op; // 0 : 等待,直到信号灯的值变成0 // 1 : 释放资源,V操作 // -1 : 分配资源,P操作 
        short sem_flg;  // 0(阻塞),IPC_NOWAIT, SEM_UNDO
         };//对某一个信号灯的操作,如果同时对多个操作,则需要定义这种结构体数组

         nops: 要操作的信号灯的个数 ,1个
    返回值:成功 :0 ; 失败:-1

      用法:semop(semid, &mysembuf, 1);
      申请资源 P操作:
        mysembuf.sem_num = 0;
        mysembuf.sem_op = -1;
        mysembuf.sem_flg = 0;
      释放资源 V操作:
        mysembuf.sem_num = 0;
        mysembuf.sem_op = 1;
        mysembuf.sem_flg = 0;

3)int semctl ( int semid, int semnum, int cmd…/*union semun arg*/);
  功能:信号灯集合的控制(初始化/删除)
  参数:semid:信号灯集ID
       semnum: 要操作的集合中的信号灯编号
       cmd:
      GETVAL:获取信号灯的值,返回值是获得值
      SETVAL:设置信号灯的值,需要用到第四个参数:共用体
      IPC_RMID:从系统中删除信号灯集合
  返回值:成功 0 ; 失败 -1

  用法:
  初始化:semctl(semid, 0, SETVAL, mysemun);
  需要在程序中定义共用体
  获取信号灯值:semctl(semid, 0, GETVAL);
  删除信号灯集:semctl(semid, 0, IPC_RMID);

例子: 简单对信号灯的操作(多值信号灯)

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <errno.h> int semid; //信号灯集id, 通过这个id 操作这个信号灯集,在不同进程中,操作同一个信号灯集的id
union semnum
{
int val;
}; union semnum mynum;
struct sembuf mybuf; //定义操作信号灯的结构 //参数 信号灯集id 和 是哪个信号灯
void sem_p(int semid, unsigned short num) //P操作函数
{
mybuf.sem_num = num; //第一个信号灯,(信号灯编号)
mybuf.sem_op = -; //进行P操作, 为 1 时表示V操作
mybuf.sem_flg = ; //阻塞
semop(semid, &mybuf, ); //最后一个参数,表示操作信号灯的个数
} //参数 信号灯集id 和 是哪个信号灯
void sem_v(int semid, unsigned short num) //V操作函数
{
mybuf.sem_num = num;
mybuf.sem_op = ; //1 表示V 操作
mybuf.sem_flg = ; //阻塞
semop(semid, &mybuf, ); //操作的 mybuf 全部变量信号灯集
} int main(int argc, const char *argv[])
{
key_t key;
key = ftok("app",'m');//获取唯一的 key
if(key < )
{
perror("fail fptk ");
exit();
}
//int semget(key_t key, int nsems, int semflg);
//IPC_EXCL | IPC_CREAT 信号灯不存在就创建
semid = semget(key, , IPC_CREAT|IPC_EXCL|); //创建信号灯,IPC_EXCL 问信号灯存不存在
if(semid < )
{
if(errno == EEXIST)//存在时,只需要打开即可
{
semid = semget(key,,); //打开信号灯
}
else
{
perror("semget fail ");
}
}
else
{
// int semctl(int semid, int semnum, int cmd, ...);
mynum.val = ; //设置信号灯值
semctl(semid,,SETVAL,mynum); //初始化 0 号信号灯,此处使用共用体设置信号灯值,初始化完成后可以继续给下一个信号灯设置值 mynum.val = ; //设置信号灯值
semctl(semid,,SETVAL,mynum); //初始化 1 号信号灯,把设置的信号灯值给1号信号灯
}
sem_p(semid, ); //对semid 指向的信号灯集,中0号信号灯进行P操作
sem_p(semid, ); //对semid 指向的信号灯集,中0号信号灯进行P操作
sem_p(semid, ); //对semid 指向的信号灯集,中0号信号灯进行P操作
printf("%d\n",semctl(semid, , GETVAL)); //上面对 0 信号灯初始值 10 进行流三次P 操作,此时应该为 7
printf("%d\n",semctl(semid, , GETVAL)); //没有对 1 信号灯操作,其值不变为 5
putchar(); sem_v(semid,);
printf("%d\n",semctl(semid, , GETVAL)); //上面对 1 信号灯 V 操作,其值变为 6 semctl(semid,,IPC_RMID);//删除信号灯
return ;
}

测试:对于多值信号灯可以进行多次P操作

Linux 进程间通信 信号灯集的更多相关文章

  1. Linux进程间通信:IPC对象——信号灯集详解

    作者:倪老师,华清远见嵌入式学院讲师. 一.信号灯概述 信号灯与其他进程间通信方式不大相同,它主要提供对进程间共享资源访问控制机制.相当于内存中的标志,进程可以根据它判定是否能够访问某些共享资源,同时 ...

  2. <转>Linux环境进程间通信--信号灯(四)

    http://www.ibm.com/developerworks/cn/linux/l-ipc/part4/ 一.信号灯概述 信号灯与其他进程间通信方式不大相同,它主要提供对进程间共享资源访问控制机 ...

  3. 【转】 Linux进程间通信

    一.进程间通信概述进程通信有如下一些目的:A.数据传输:一个进程需要将它的数据发送给另一个进程,发送的数据量在一个字节到几M字节之间B.共享数据:多个进程想要操作共享数据,一个进程对共享数据的修改,别 ...

  4. (转载)Linux进程间通信

    (在学习linux进程通信,看到一篇很好的文章,转载过来,原文地址是http://www.cnblogs.com/linshui91/archive/2010/09/29/1838770.html) ...

  5. Linux进程间通信--进程,信号,管道,消息队列,信号量,共享内存

    Linux进程间通信--进程,信号,管道,消息队列,信号量,共享内存 参考:<linux编程从入门到精通>,<Linux C程序设计大全>,<unix环境高级编程> ...

  6. linux进程间通信-(转自 临水)

    一.进程间通信概述进程通信有如下一些目的:A.数据传输:一个进程需要将它的数据发送给另一个进程,发送的数据量在一个字节到几M字节之间B.共享数据:多个进程想要操作共享数据,一个进程对共享数据的修改,别 ...

  7. Linux进程间通信(一): 信号 signal()、sigaction()

    一.什么是信号 用过Windows的我们都知道,当我们无法正常结束一个程序时,可以用任务管理器强制结束这个进程,但这其实是怎么实现的呢?同样的功能在Linux上是通过生成信号和捕获信号来实现的,运行中 ...

  8. Linux进程间通信(二):信号集函数 sigemptyset()、sigprocmask()、sigpending()、sigsuspend()

    我们已经知道,我们可以通过信号来终止进程,也可以通过信号来在进程间进行通信,程序也可以通过指定信号的关联处理函数来改变信号的默认处理方式,也可以屏蔽某些信号,使其不能传递给进程.那么我们应该如何设定我 ...

  9. linux进程间通信-概述

    一 进程间通信有如下的目的: 1.数据传输,一个进程需要将它的数据发送给另一个进程,发送的数据量在一个字节到几M之间:2.共享数据,多个进程想要操作共享数据,一个进程对数据的修改,其他进程应该立刻看到 ...

随机推荐

  1. tushare积分怎么获得 tushare pro 积分充值 积分转让

    本人是做量化投资的,团队转型,换了交易策略,手头有多个离职同事的闲置转让.600分:原价50元,仅需39元1500分:原价150元,仅需109元(售罄)2000分:原价200元,仅需149元5000分 ...

  2. 在python中创建列表的最佳和/或最快方法

    在python中,据我所知,至少有3到4种方法来创建和初始化给定大小的列表: 简单循环append: my_list = [] for i in range(50): my_list.append(0 ...

  3. 关于shell脚本中的别名问题

    在shell脚本中,shell中的alias别名是不会起作用的,在脚本中的命令都是按着环境变量PATH直接找到命令文件而执行的,所以就不用担心脚本里的命令会与shell中的个性别名冲突啦~

  4. (转)Unity Cinemachine插件,实现单目标和多目标之间切换

    Unity Cinemachine插件学习笔记,实现单目标和多目标之间切换*版本要求Unity2017.1及以上. 参考资料: [官方] Unity 2017.1正式版发布 Cinemachine插件 ...

  5. 在阅读众多的blog中,我学到了什么

    写博客的人,自然会读别人的博客:读博客的人,不一定会写博客.但是这两种人之间的差别是很大的 在最近在一段时间,发现了一个好的博客,通过该博客的友链,发现了新大陆.... 从Jeff Wong开始,到老 ...

  6. master挂了的话pm2怎么处理 使用pm2方便开启node集群模式

    本文为转载 Introduction As you would probably know, Node.js is a platform built on Chrome's JavaScript ru ...

  7. js中如何避免动态引入重复资源

    创建存储数据的数组或者对象: 每次调用方法的时候,往里面添加资源对象,包括路径: 每次调用的时候遍历此路劲是否存在,如存在,就调用此资源对象的promise进行操作. 可避免资源未加载完成就执行的情况 ...

  8. C之输入输出函数(1) -- fgets()

    https://www.ibm.com/support/knowledgecenter/en/ssw_ibm_i_71/rtref/fgets.htm #include <stdio.h> ...

  9. Delphi GlobalAlloc、GlobalLock、GlobalUnlock、GlobalFree 函数

    GlobalAlloc 函数 分配一块内存,该函数会返回分配的内存句柄. GlobalLock 函数 锁定内存块,该函数接受一个内存句柄作为参数,然后返回一个指向被锁定的内存块的指针. 您可以用该指针 ...

  10. 简单的使用Qt的QCustomplot画图

    一.新建一个widget工程 二.将Qcustomplot文件复制到项目下 三.右键qt项目增加已存在的文件 四.在.pro中添加 QT += widgets printsupport 五.在.h中添 ...