美文网首页
XSI IPC之信号量集

XSI IPC之信号量集

作者: 静倚晴窗笑此生 | 来源:发表于2019-03-27 19:39 被阅读0次

    信号量集

    man semop 查看信号量集操作

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

    semget(2)

    实例化信号量集合

    函数声明

    #include <sys/types.h>
    #include <sys/ipc.h>
    #include <sys/sem.h>
    
    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 <sys/types.h>
    #include <sys/ipc.h>
    #include <sys/sem.h>
    
    int semctl(int semid, int semnum, int cmd, ...);
    

    参数含义

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

    第二个参数semnum..

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

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

    semop(2)

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

    函数声明

    #include <sys/types.h>
    #include <sys/ipc.h>
    #include <sys/sem.h>
    
    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 <stdio.h>
    #include <stdlib.h>
    #include <sys/wait.h>
    #include <sys/types.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    #include <sys/ipc.h>
    #include <sys/sem.h>
    #include <unistd.h>
    #include <string.h>
    
    #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之信号量集

          本文链接:https://www.haomeiwen.com/subject/irvtbqtx.html