信号量集用于对存在竞争的资源加锁

1.semId=semget(key,nsems,semflg)

  key:为信号量集名称,可以指定为0455等数字,也可以为PC_PRIVATE

  nsems:创建几个信号量

  semflg:创建并给权限,如(IPC_CREAT | 0600);

2.semctl(semId,semnum,cmd,...)//

  2.1初始化信号量集  

  semctl(semId,num,SETVAL,1)  将semId的第num个信号量设置为1;semvalue

  2.2删除信号量集

  semctl(semId,0,IPC_RMID)    删除信号量集semId

  .......

3. 锁的操作

  semop(semId,&buf,1)/////能做则做,不能做则阻塞等待

  锁操作是将要操作的信号放到一个结构体中 struct sembuf   buf;

  buf.sem_num:对第几个信号操作

  buf.sem_op:  有三种情况  大于0(释放资源)   小于0(申请资源)     等于0(等0操作)

  buf.sem_flg:  是否阻塞

  semop(semId,&buf,1);操作1个这样的buf

·

  semtimeop()//能做则做,不能做则阻塞一段时间后返回,常用于超时等待

代码示例

//等零操作:子进程执行10次 父进程执行一次
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <sys/wait.h>

#define      NR     1

int p_lock(int semId);
void v_unlock(int semId);
void w_zero(int semId);

int main(void)
{
    int semId,ret;
    pid_t pid;

semId=semget(IPC_PRIVATE,NR,0600);
    if(semId==-1)
    {
        perror("semget");
        return 1;
    }
    //init sem value
    ret=semctl(semId,0,SETVAL,0);
    if(ret==-1)
    {
        perror("set sem val");
        semctl(semId,0,IPC_RMID);
        return 2;
    }

//create child process  继承semId
    pid=fork();
    if(pid==-1)
    {
        semctl(semId,0,IPC_RMID);
        perror("fork");
        return 3;
    }
    else if(pid==0)//child
    {
        int count=1;
        while(1)
        {
            p_lock(semId);
            printf("child do %dth\n",count);
            count++;
            sleep(1);
        }
        exit(0);
    }
    //aprent
    while(1)
    {
        w_zero(semId);
        printf("parent do....\n");

v_unlock(semId);
    }

wait(NULL);
 
    semctl(semId,0,IPC_RMID);
    return 0;
}
int p_lock(int semId)
{
    struct sembuf  buf[1];
    buf[0].sem_num=0;//第几个
    buf[0].sem_op =-1;//怎么操作(1.op>0 指释放资源   2.op<0 申请资源  3.op==0等大资源为0)
    buf[0].sem_flg=0; //IPC_NOWAIT

return semop(semId,buf,1);
}

void v_unlock(int semId)
{
    struct sembuf  buf;
    buf.sem_num=0;//第几个
    buf.sem_op =10;//怎么操作(1.op>0 指释放资源   2.op<0 申请资源  3.op==0等大资源为0)
    buf.sem_flg=0; //IPC_NOWAIT

semop(semId,&buf,1);
}
void w_zero(int semId)
{
    struct sembuf  buf;
    buf.sem_num=0;//第几个
    buf.sem_op =0;//怎么操作(1.op>0 指释放资源   2.op<0 申请资源  3.op==0等大资源为0)
    buf.sem_flg=0; //IPC_NOWAIT

semop(semId,&buf,1);

}

IPC之——信号量集的更多相关文章

  1. IPC之——信号量集(多个信号量)

    如果两个进程不仅需要同步,还要保证先后执行顺序,就要用两个信号量(互斥锁)来解决 //栅栏模型:实现以下框架中的四个子进程 所有进程做完任务后 在一起执行下一次  #include <stdio ...

  2. System V IPC 之信号量

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

  3. Linux IPC 之信号量

    信号量(也叫信号灯)是一种用于提供不同进程间或一个给定进程的不同线程间同步手段的原语. 信号量是进程/线程同步的一种方式,有时候我们需要保护一段代码,使它每次只能被一个执行进程/线程运行,这种工作就需 ...

  4. IPC之信号量

    无名信号量 POSIX标准提出了有名信号量和无名信号量来同步进程和线程,而linux(2.6以前)只实现了无名信号量. sem_overview中有详细介绍:man 7 sem_overview. S ...

  5. IPC进程间通信---信号量

    信号量 信号量:信号量是一个计数器,常用于处理进程或线程的同步问题,特别是对于临界资源访问的同步.临界资源可以 理解为在某一时刻只能由一个进程或线程操作的资源,这里的资源可以是一段代码.一个变量或某种 ...

  6. linux IPC的信号量

    信号量相关函数原型 获得一个信号量ID #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h&g ...

  7. ucos信号量集源码分析

    在实际的应用之中,一个任务经常需要等待多个信号量的同时生效,或者说任务需要根据多个信号量的组合作用的结果来决定任务的运行方式,为了实现这种多信号量组合的功能,ucos实现了信号量集的特殊结构. 信号量 ...

  8. μC/OS-II 信号量集

    简介 在实际应用中,任务常常需要与多个事件同步,即要根据多个信号量组合作用的结果来决定任务的运行方式.μC/OS-II 为了实现多个信号量组合的功能定义了一种特殊的数据结构--信号量集. 信号量集所能 ...

  9. 【iCore4 双核心板_uC/OS-II】例程十:信号量集

    一.实验说明: 在实际应用中,任务常常需要与多个事件同步,即要根据多个信号量组合作用的结果来决定任务的运行方式.UCOSII为了实现多个信号量组合的功能定义了一种特殊的数据结构——信号量集. 二.实验 ...

随机推荐

  1. C++多态性与虚函数

    派生一个类的原因并非总是为了继承或是添加新的成员,有时是为了重新定义基类的成员,使得基类成员“获得新生”.面向对象的程序设计真正的力量不仅仅是继承,而且还在于允许派生类对象像基类对象一样处理,其核心机 ...

  2. Windb实践之Script Command

    1.输出参数 .echo The first argument is ${$arg1}. .echo The fifth argument is ${$arg5}. .echo The fourth ...

  3. 创建简单spring boot项目

    简介 使用spring boot可以轻松创建独立的,基于Spring框架的生产级别应用程序.Spring boot应用程序只需要很少的spring配置 特点 创建独立的Spring应用程序 直接嵌入t ...

  4. delphi 单例模式

    unit Singleton; (* 单例模式适用于辅助类, 一般伴随于单元的生命周期 *) interface uses SysUtils; type TSingleton = class publ ...

  5. 吴裕雄--天生自然TensorFlow2教程:Broadcasting

    Broadcasting可以理解成把维度分成大维度和小维度,小维度较为具体,大维度更加抽象.也就是小维度针对某个示例,然后让这个示例通用语大维度. import tensorflow as tf x ...

  6. PAT Advanced 1147 Heaps (30) [堆,树的遍历]

    题目 In computer science, a heap is a specialized tree-based data structure that satisfies the heap pr ...

  7. Docker MongoDB 集群搭建

    简单地在Docker环境上搭建一个无认证的MongoDB集群.1.本文使用的容器集群角色 ContainerName IP:portConfig Server cfg_1    10.1.1.2:27 ...

  8. Unity使用TUIO协议接入雷达

    本篇文章不介绍Unity.TUIO.雷达是什么以及有什么作用.刚接触TUIO的亲们,建议直接硬刚.至于刚接触Unity的亲,这边建议亲直接放弃治疗呢 下面开始正儿八经的教程 需要准备的东西 Unity ...

  9. 从定时器的选型,到透过源码看XXL-Job(下)

    透过源码看xxl-job (注:本文基于xxl-job最新版v2.0.2, quartz版本为 v2.3.1. 以下提到的调度中心均指xxl-job-admin项目) 上回说到,xxl-job是一个中 ...

  10. vue 常用知识点

    1.数据变更,页面渲染完成 this.$nextTick(function(){ alert('v-for渲染已经完成') }) 2.iview select组件 setQuery用法 <Sel ...