Linux---进程间通信IPC之信号量

进程间通信(IPC):是指在不同进程之间传播或交换信息。
IPC的方式:通常有管道(无名管道、命名管道)、消息队列、信号量、共享存储、Socket、Streams等(Socket和Streams支持不同主机上的两个进程IPC)
进程间通信的目的:
1.数据传输:一个进程需要将它的数据发给另一个进程
2.资源共享:多个进程之间共享同样的资源
3.通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发送了啥(如进程终止要通知父进程)
4.进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变

SYSTEM V信号量,这是信号量值的集合,不是单个的信号量,相关的信号量操作函数由“sys/ipc.h”引用,这篇主要描述了这个信号量集
信号量
也称信号灯,主要用来控制多个进程对共享资源的访问,信号量是进程通信的一种重要方法,但其本身并不进行数据交换,和管道消息队列不同,而是以同步,互斥,协调于各个进程之间,协调进程对共享资源的访问,共享内存的使用就要用到信号量。把多个执行流能够访问到的资源叫做公共资源,把公共资源的代码叫做临界区

当进程不再使用一个信号量控制的共享资源时,信号量+1,对信号量的值进行的增减均为原子操作
这是由于信号量的主要作用是维护资源的互斥或所线程的同步访问
但在信号量的创建及初始化上,不能保证操作均为原子性

信号量本身就是个计数器,计数临界资源的剩余数量
如果一个进程访问临界资源,步骤:

  • 监控控制该资源的信号量
  • 如果信号量>0,则访问该资源,并将信号量-1
  • 如果信号量的值<=0,该进程被挂起,直到信号量>0,即有了空闲资源时才被唤醒,返回到第一步
  • 进程使用完成临界资源,将信号量+1,这时如果有其他进程正在等待资源,将其唤醒

内核为每个信号量集维护了一个信号量结构体,在sys/sem.h

struct semid_ds
{
    struct ipc_perm sem_perm;//信号量集的操作权限
    struct sem* sem_base;//信号量结构数组指针,
    ushort sem_nsems;//数组的个数,即信号量的个数
    time_t sem_otime;//最近一次修改的时间
    time_t sem_ctime;//创建成功的时间
}
struct sem
{
    ushort semval;//信号量的值
    short sempid;//最近一次返回该信号量的进程ID
    ushort semncat;//等待资源的进程数
    ushort semzcnt;//正在使用资源的进程数
}

创建和打开信号量集

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

 - nsems>0创建新的信号量集里信号量的个数,创建后不能修改,nsems==0打开信号量集
 - 返回值:信号量标识符semid,可以唯一标识信号量集
 - sem_perm的uid gid被设置成有效的
 - oflag存入sem_perm.mode
 - semget函数执行成功后,产生一个由内核维持的类型为semid_ds结构体的信号量集

对信号量集的控制

int semctl(int semid,int _semum,int cmd,...)
union semum
{
    int val;//信号量的值,SETVAL
    struct semid_ds* buf;//信号量的状态,GETVAL、GETPID、GETZCNT
    unsigned short* array;//所有信号量的值
    struct seminfo* _buf;//IPC_INFO使用的缓存区
}
 - semum是信号在集合中的序号
 - cmd,控制命令
    IPC_RMID 删除信号量
    IPC_EXCL 只有在信号量集不存在时才创建
    IPC_SET 设置访问权限
    SETVAL 设置信号量的值
    GETVAL 获取信号量的值
    GETPID 获取最近操作信号量的进程ID
    GETNCNT 获取等待资源的进程数
    GETZCNT  获取占用资源的进程数

对信号量的操作PV

int semop(int semid,struct sembuf* opsptr,size_t nops);
struct sembuf
{
    short sem_num;//信号量的编号
    short sem_op;//信号量的操作
    short sem_flg;//操作标识符
}
 - opsptr 指向信号量结构数组
 - nops opst指向的数组中的结构体的个数,即信号量的个数
 - sem_op 正数,其值加到semval上,释放信号量控制的资源数
     sem_op 负数,其值减到semval上
 - sem_flg
     SEM_UNDO 如果在PV操作上设置了这个选项,一旦进程异常,就将信号量恢复成操作前的样子
     IPC_NOWAIT 不阻塞
union semum
{
    int val;
    struct semid_ds *buf;
    unisgned short *array;
    struct seminfo *_buf;
};
ststic int commSemSet(int nums)
{
    key_t _k=ftok(PATHnME,PROJ_ID);//key值用来保证看到的是同一信号量集
    if(_k<0)
    {
        perror("ftok");
        return -1;
    }
    int semid=semget(_k,nums,IPC_CREAT|IPC_EXCL);
    if(semid<0)
    {
        perror("semget");
        return -2;
    }
    return semid;
}
int creatSemSet(int nums)
{
    return commSemSet(nums,IPC_CREAT|IPC_EXCL);
}
int getSemSet()
{
    return commSemSet(nums,IPC|CREAT);
}
int InitSemSet()
{
    union semun _un;
    _un.val=val;
    if(semct(semid,which,SETVAL,_un)!=0)
    {
        perror("semctl");
        return -1;
    }
    return 0;
}
int destorySemSet(int semid)
{
    if(semctl(semid,0,IPC|RMID)<0)
    {
        perror("semctl");
        return -1;
    }
    retnurn 0;
}
static int commPV(int semid,int num,int op)
{
    struct sembuf _sf;
    _sf.sem_num=num;
    _sf.sem_op=-1;
    _sf.sem_flg=0;   //SEM_UNDO如果在PV操作上设置了这个选项,一旦进程异常,就将信号量恢复成操作前的样子
    if(semop(semid,&_sf,1)!=0)
    {
        perror("semop");
        return -1;
    }
    return 0;
}
int P(int semid,int num)
{
    return commPV(semid,num, -1);
}
int V(int semid,int num)
{
    return commPV(semid,num, 1);
}

如果有什么不对的地方,可以评论告诉我,望指导!

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值