信号量概念及初级编程实战

标题信号量概念及初级编程实战

1相关概念:

以一个停车场的运作为例。简单起见,假设停车场只有三个车位,一开始三个车位都是空的。这时如果同时来了五辆车,看门人允许其中三辆直接进入,然后放下车拦,剩下的车则必须在入口等待,此后来的车也都不得不在入口处等待。这时,有一辆车离开停车场,看门人得知后,打开车拦,放入外面的一辆进去,如果又离开两辆,则又可以放入两辆,如此往复。

在这个停车场系统中,车位是公共资源,每辆车好比一个线程,看门人起的就是信号量的作用。信号量等同于钥匙,只有获取了钥匙才能进入进程,同时在退出进程后,又必须将钥匙放回到原处,即p操作和v操作。

2相关api的介绍

a: semget获取创建信号量api:
int semget(key_t key, int nsems, int semflg);

b: semctl初始化信号量:
//int semctl(int semid, int semnum, int cmd);

c:拿走放回钥匙api semop:
//int semop(int semid, struct sembuf *sops, unsigned nsops);

3初级父子进程间信号量操作编程实战:

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/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) */
};


void pGetKey(int id)//int semop(int semid, struct sembuf *sops, unsigned nsops);
{
        struct sembuf set;

        set.sem_num=0;

        set.sem_op =-1;

        set.sem_flg=SEM_UNDO;

        semop(id,&set,1);

        printf("get key\n");
}


void vPutBackKey(int id)//
{
        struct sembuf set;

        set.sem_num=0;

        set.sem_op =1;

        set.sem_flg=SEM_UNDO;

        semop(id,&set,1);

        printf("put back the key\n");
}
int main()
{
        key_t key;

        int semid;

        key=ftok(".",1);
                                   //信号量集合中有一个信号量
        semid=semget(key,1,IPC_CREAT|0666);//获取,创建信号量

        union semun initsem;
                                //操作第0个信号量
        initsem.val=0;

        semctl(semid,0,SETALL,initsem);//初始化信号量
                                        //设置信号量的值,设置为initsem
        int pid=fork();

        if(pid>0){
                pGetKey(semid);

                printf("this is father\n");

                vPutBackKey(semid);
        }else if(pid==0){
                vPutBackKey(semid);

                printf("this is child\n");
        }else{
                printf("creat pid failuer\n");
        }

        printf("wencai hen shuai");

        return 0;
        
}

让我们来看看实际的运行效果:

在这里插入图片描述
由结果来看,可以看出子进程运行在父进程的前面,一般父子进程的先后我们是未知的,但通过信号量的这种使用方式,我们就可以控制这两个进程的先后顺序。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值