进程间通信——信号量

1.定义

信号量是一个特殊的变量,程序对其访问都是原子操作,且只允许对它进行等待(即P(信号变量))和发送(即V(信号变量))信息操作。最简单的信号量是只能取0和1的变量,这也是信号量最常见的一种形式,叫做二进制信号量。而可以取多个正整数的信号量被称为通用信号量。这里主要讨论二进制信号量。

2.P、V操作

P、V操作是定义在信号量上的S上的两个操作
P(S)
a、 S:=S-1;
b、若S>=0,则调用P(S)的进程继续运行;
c、若S<0,则调用P(S)的进程被阻塞,并把它插入到等待信号量S的阻塞队列中。
V(S)
a、S:=S+1;
b、若S>0,则调用V(S)的进程继续运行;
c、若S<=0,从等待信号量S的阻塞队列中唤醒头一个进程,然后调用V(S)的进程继续运
3.函数
int semget(key_t key,int num_sems,int sem_flags);
作用:创建一个新信号量或取得一个已有信号量的键
返回值:成功返回信号量ID,出错返回-1。
key_t key:为整型值,用户可以自己设定。
int num_sems:初始化信号量的个数。
int sem_flags:信号量的创建方式或权限。有IPC_CREAT,IPC_EXCL。
IPC_CREAT如果信号量不存在,则创建一个信号量,否则获取。
IPC_EXCL只有信号量不存在的时候,新的信号量才建立,否则就产生错误。

ipcs -s 查看信号量是否创建成功
ipcrm -s simid 删除指定的信号量

int semop(int semid,struct *sem_ops , size_t num_sem_ops)
作用:用户改变信号量的值,即使用资源还是释放资源的使用权。
返回值:成功返回0,失败返回-1。
int semid:信号量的标识码,即senget()函数的返回值。
struct *sem_ops:指向结构体数组的指针。
struct sembuf
{
struct short sem_num;//在信号量集中的编号,第一个为0
short sem_op;//对信号量的操作
short sem_flg;//通常被设置为SEM_UNDO,程序结束时(不论正常或不正常),保证信号值会被重设为semop()调用前的值。这样做的目的在于避免程序在异常情况下结束时未将锁定的资源解锁,造成该资源永远锁定。
};
size_t num_sem_ops:nsops:操作结构的数量,恒大于或等于1。

int semctl(int sem_id, int sem_num, int command,…….);
作用:控制信号量信息。
返回值:成功返回0,失败返回-1。
int sem_id: 信号量的标志码(ID),也就是semget()函数的返回值;
int sem_num:操作信号在信号集中的编号。从0开始。
int command: 命令,表示要进行的操作。

参数command中可以使用的命令如下:
IPC_STAT读取一个信号量集的数据结构semid_ds,并将其存储在semun中的buf参数中。
IPC_SET设置信号量集的数据结构semid_ds中的元素ipc_perm,其值取自semun中的buf参数。
IPC_RMID将信号量集从内存中删除。
GETALL用于读取信号量集中的所有信号量的值。
GETNCNT返回正在等待资源的进程数目。
GETPID返回最后一个执行semop操作的进程的PID。
GETVAL返回信号量集中的一个单个的信号量的值。
GETZCNT返回这在等待完全空闲的资源的进程数目。
SETALL设置信号量集中的所有的信号量的值。
SETVAL设置信号量集中的一个单独的信号量的值。

若有第四个参数,它是一个联合体结构:
union semun
{
int val;
struct semid_ds *buf;
unsigned short *arrary;
};

代码:

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

int semid;
union semun    //semctl的第四个参数,它是一个联合体
{
    int val;
};


//由进程调用,确保获取到信号量集
void sem_get(int key,int val)
{
    semid=semget((key_t)key,1,0666);//semid是信号量的标志码
    if(semid==-1)//没有此信号量集
    {
        semid=semget((key_t)key,1,0666|IPC_CREAT);//在内核中创建信号量集
        assert(semid!=-1);    
        union semun v;      //semctl的第四个参数是一个联合体结构
        v.val=val;
        semctl(semid,0,SETVAL,v);
    }
}
//p操作
void sem_p()
{
    struct sembuf op;//semop的第二个参数是指向结构体数组的指针
    op.sem_num=0;
    op.sem_op=-1;
    op.sem_flg=SEM_UNDO;//若进程没有释放该信号量,则操作系统自动释放
    semop(semid,&op,1);  

}
//v操作
void sem_v()
{
    struct sembuf op;
    op.sem_num=0;
    op.sem_op=1;
    op.sem_flg=SEM_UNDO;
    semop(semid,&op,1);
}

void sem_del()
{
    semctl(semid,0,IPC_RMID,NULL);
}

mainA.c

#include"sem.h"

void main()
{
    int fd=open("FIFO",O_WRONLY);
    assert(fd!=-1);
    sem_get(1234,0);
    while(1)
    {
        printf("please input:");
        fflush(stdout);
        char buff[128]={0};
        fgets(buff,128,stdin);
        buff[strlen(buff)-1]=0;
        write(fd,buff,strlen(buff));
        sem_v();

    }
}

mainB.c

#include"sem.h"

void main()
{
    int fd=open("FIFO",O_RDONLY);
    assert(fd!=-1);
    sem_get(1234,0);
    while(1)
    {
        sem_p();
        printf("B proc start\n");
        char buff[128]={0};
        read(fd,buff,127);
        printf("B ::%s\n",buff);
    }
}

这里写图片描述

“`

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
信号量是一种用于进程间通信和同步的机制。它是一个计数器,用于保证在共享资源上的互斥访问。在Linux系统中,可以使用信号量来实现进程间的同步和互斥。以下是信号量的基本概念: - 计数器:信号量的值是一个计数器,它可以被多个进程共享。 - P操作:当一个进程需要访问共享资源时,它必须执行P操作,该操作会将信号量的值减1。如果信号量的值为0,则进程将被阻塞,直到信号量的值大于0。 - V操作:当一个进程使用完共享资源后,它必须执行V操作,该操作会将信号量的值加1。如果有进程正在等待该信号量,则唤醒其中一个进程继续执行。 在ZUCC中,可以使用信号量来实现进程的同步和互斥。首先,需要使用semget函数创建一个信号量集合,并使用semctl函数对信号量进行初始化。然后,可以使用semop函数执行P和V操作。例如,下面是一个简单的示例程序,用于演示信号量的使用: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/sem.h> #define SEM_KEY 1234 union semun { int val; struct semid_ds *buf; unsigned short *array; }; int main() { int semid, pid; union semun arg; struct sembuf sb; // 创建信号量集合 semid = semget(SEM_KEY, 1, IPC_CREAT | 0666); if (semid == -1) { perror("semget"); exit(EXIT_FAILURE); } // 初始化信号量 arg.val = 1; if (semctl(semid, 0, SETVAL, arg) == -1) { perror("semctl"); exit(EXIT_FAILURE); } // 创建子进程 pid = fork(); if (pid == -1) { perror("fork"); exit(EXIT_FAILURE); } else if (pid == 0) { // 子进程执行P操作 sb.sem_num = 0; sb.sem_op = -1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop P"); exit(EXIT_FAILURE); } printf("Child process\n"); // 子进程执行V操作 sb.sem_num = 0; sb.sem_op = 1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop V"); exit(EXIT_FAILURE); } exit(EXIT_SUCCESS); } else { // 父进程执行P操作 sb.sem_num = 0; sb.sem_op = -1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop P"); exit(EXIT_FAILURE); } printf("Parent process\n"); // 父进程执行V操作 sb.sem_num = 0; sb.sem_op = 1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop V"); exit(EXIT_FAILURE); } exit(EXIT_SUCCESS); } return 0; } ``` 在上述代码中,创建了一个信号量集合,并将其初始化为1。然后,创建了一个子进程和一个父进程,它们分别执行P和V操作。由于信号量的初始值为1,因此父进程和子进程都可以顺利地执行。如果将信号量的初始值改为0,那么父进程和子进程都将被阻塞,直到有一个进程执行V操作为止。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值