XSI IPC之信号量集

信号量集

man semop 查看信号量集操作

信号量集:进程进程间发送信号,是一个大容器,容器中有多个元素, 每个元素有多个信号

semget(2)

实例化信号量集合

函数声明

#include 
#include 
#include 

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

参数含义

第一个参数为,ftok返回的key_t类型,如果 key值为 IPC_PRIVATE 则只能用于有亲缘关系

标识符可选(多选用或运算)

第二个参数nsems是该集合中信号量个数,如果创建新的集合,必须指定nsems,如果是引用现有的集合,则将nsems指定为0;

第三个参数semflg可选参数:IPC_CREAT 创建,需要或上文件权限

IPC_EXCLIPC_CREAT一起使用,确保如果消息队列已经创建,则返回失败

返回值

成功返回信号量集ID号

失败返回 -1 并且设置errno 如果标识符使用了IPC_CREATIPC_EXCL 并且errno的值为EEXIST,代表信号量集已经存在。

semctl(2)

第三个参数为以下两个选项时:

SETVAL 初始化指定集合中每个成员信号的个数

IPC_RMID 销毁信号量集

函数声明

#include 
#include 
#include 

int semctl(int semid, int semnum, int cmd, ...);

参数含义

第一个参数semid为信号量集ID,即semget(2)返回值

第二个参数semnum..

第三个参数cmd为可选项SETVAL : 初始化指定集合中每个成员信号的个数;IPC_RMID :销毁信号量集

第四个参数可选,取决于cmd参数

semop(2)

自动执行信号量集合上的操作数组

函数声明

#include 
#include 
#include 

int semop(int semid, struct sembuf *sops, size_t nsops);

参数含义

sops是一个指针,指向一个struct sembuf结构表示的信号量操做数组:

struct sembuf
{
  unsigned short sem_num;  /* semaphore number */
  short          sem_op;   /* semaphore operation */
  short          sem_flg;  /* operation flags */
}

信号量集示例

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#define FLNAME  "/tmp/out"

static int semid;

static int child_job(void);
int main(void)
{
    int fd;
    pid_t pid;

    fd = open(FLNAME, O_WRONLY | O_CREAT | O_TRUNC, 0666);
    if (fd < 0) {
        perror("open()");
        exit(1);
    }
    write(fd, "0", 1);
    close(fd);

    // 实例化信号量集合
    semid = semget(IPC_PRIVATE, 1, IPC_CREAT | 0600);
    if (semid == -1) {
        perror("semget()");
        exit(1);
    }

    // 向信号量中初始化信号的个数
    semctl(semid, 0, SETVAL, 1);

    for (int i = 0; i < 20; i++) {
        pid = fork();   
        if (pid < 0) {
            perror("fork()");
            goto FORK_ERROR;
        }
        if (pid == 0) {
            // 
            child_job();    
            exit(0);
        }
    }   

    for (int i = 0; i < 20; i++)
        wait(NULL);

    exit(0);
FORK_ERROR:
    exit(1);
}

static int P(void)
{
    struct sembuf buf;

    buf.sem_num = 0;
    buf.sem_op = -1;
    buf.sem_flg = 0;
    if (semop(semid, &buf, 1) == -1) {
        perror("semop()");
        return -1;
    }

    return 0;
}

// 还
static int V(void)
{
    struct sembuf buf;

    buf.sem_num = 0;
    buf.sem_op = 1;
    buf.sem_flg = 0;
    if (semop(semid, &buf, 1) == -1) {
        perror("semop()");
        return -1;
    }

    return 0;
}

static int child_job(void)
{
    FILE *fp = NULL;
    char buf[100] = {};

    fp = fopen(FLNAME, "r+");
    if (NULL == fp) {
        perror("fopen()");
        return -1;
    }
    
    // 拿信号
    P();

    fgets(buf, 100, fp);
    // sleep(1);
    rewind(fp);
    fprintf(fp, "%d", atoi(buf)+1);
    fclose(fp);

    // 还信号
    V();

    return 0;
}

你可能感兴趣的:(XSI IPC之信号量集)