当前位置: 首页 > news >正文

信号灯集,消息队列

信号灯集

1、概念

信号灯(semaphore),也叫信号量。它是不同进程间或一个给定进程内部不同线程间同步的机制;System V的信号灯是一个或者多个信号灯的一个集合。其中的每一个都是单独的计数信号灯。而Posix信号灯指的是单个计数信号灯。

通过信号灯集实现共享内存的同步操作。

  1. 步骤

在不同的进程间,通过相同的key值,打开相同的信号灯集

  1. 创建key值 ftok
  2. 创建或打开信号灯集semget
  3. 初始化信号灯 semctl
  4. PV操作 semop
  5. 删除信号灯集 semctl
  1. 函数

1)semget 创建\打开信号灯

int semget(key_t key, int nsems, int semflg);
功能:创建/打开信号灯
参数:key:ftok产生的key值
    nsems:信号灯集中包含的信号灯数目
    semflg:信号灯集的访问权限,通常为IPC_CREAT |IPC_EXCL |0666
返回值:成功:信号灯集ID
       失败:-1

2)semctl 信号灯集合的控制(初始化/删除)

int semctl ( int semid, int semnum,  int cmd…/*union semun arg*/);
功能:信号灯集合的控制(初始化/删除)
参数:semid:信号灯集ID
    semnum: 要操作的集合中的信号灯编号
     cmd:
        GETVAL:获取信号灯的值,返回值是获得值
        SETVAL:设置信号灯的值,需要用到第四个参数:共用体
        IPC_RMID:从系统中删除信号灯集合
返回值:成功 0
      失败 -1
用法:初始化:
union semun{int val;
}mysemun;
mysemun.val = 10;
semctl(semid, 0, SETVAL, mysemun);
获取信号灯值:函数semctl(semid, 0, GETVAL)的返回值
删除信号灯集:semctl(semid, 0, IPC_RMID);

3)semop 对信号灯集合中的信号量进行PV操作

int semop ( int semid, struct sembuf  *opsptr,  size_t  nops);
功能:对信号灯集合中的信号量进行PV操作
参数:semid:信号灯集ID
     opsptr:操作方式
     nops:  要操作的信号灯的个数 1
返回值:成功 :0
      失败:-1
struct sembuf {short  sem_num; // 要操作的信号灯的编号short  sem_op;  //    0 :  等待,直到信号灯的值变成0//   1  :  释放资源,V操作//   -1 :  申请资源,P操作                    short  sem_flg; // 0(阻塞),IPC_NOWAIT, SEM_UNDO
};
用法:
申请资源 P操作:
    mysembuf.sem_num = 0;
    mysembuf.sem_op = -1;
    mysembuf.sem_flg = 0;semop(semid, &mysembuf, 1);
释放资源 V操作:
    mysembuf.sem_num = 0;
    mysembuf.sem_op = 1;
    mysembuf.sem_flg = 0;semop(semid, &mysembuf, 1);

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <errno.h>union semun {int val;
};int main(int argc, char const *argv[])
{//1.创建key值
    key_t key = ftok(".", 'x');if (key < 0){perror("ftok err");return -1;}printf("key:%d\n", key);//2.创建或打开信号灯集int semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0666); //创建if (semid <= 0)                                          //创建失败{if (errno == 17)
            semid = semget(key, 2, 0666); //打开else{perror("semget err");return -1;}}else //创建成功-->初始化{//初始化//创建初始化union semun sem;
        sem.val = 0;semctl(semid, 0, SETVAL, sem); //0号灯初值0
        sem.val = 10;semctl(semid, 1, SETVAL, sem); //1号灯初值10}printf("semid:%d\n", semid);printf("%d\n", semctl(semid, 0, GETVAL));printf("%d\n", semctl(semid, 1, GETVAL));//PV操作struct sembuf buf;// 操作0号灯
    buf.sem_num = 0; //信号灯得编号
    buf.sem_op = 1;  //释放资源 : +1
    buf.sem_flg = 0;//1:操作1个灯semop(semid, &buf, 1);printf("%d\n", semctl(semid, 0, GETVAL));//操作1号灯
    buf.sem_num = 1; //信号灯得编号
    buf.sem_op = -1; //申请资源:-1
    buf.sem_flg = 0; //当信号灯得资源为0时,申请不到资源,阻塞//IPC_NOWAIT:到0不阻塞//1:操作一个灯semop(semid, &buf, 1);printf("%d\n", semctl(semid, 1, GETVAL));//删除信号灯集semctl(semid, 0, IPC_RMID);return 0;
}

  1. 命令查看灯集

ipcs -s:查看信号灯集

ipcrm -s semid:删除信号灯集

  1. 练习

两个进程实现通信,一个进程循环从终端输入,另一个进程循环打印,当输入quit时结束

共享内存+信号灯集+封装函数(自主选择)

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <errno.h>
#include <string.h>
#include <unistd.h>union semun {int val;
};int main(int argc, char const *argv[])
{//创建key
    key_t key = ftok(".", 'a');if (key < 0){perror("ftok err");return -1;}printf("key:%d\n", key);//创建或打开共享内存int shmid = shmget(key, 256, IPC_CREAT | IPC_EXCL | 0666);if (shmid < 0){if (errno == 17)
            shmid = shmget(key, 256, 0666);else{perror("shmget err");return -1;}}printf("shmid:%d\n", shmid);//映射char *= shmat(shmid, NULL, 0);if (== (char *)-1){perror("shmat err");return -1;}//创建或打开信号灯集int semid = semget(key, 1, IPC_CREAT | IPC_EXCL | 0666);if (semid <= 0){if (errno == 17)
            semid = semget(key, 1, 0666);else{perror("semget err");return -1;}}else{//初始化union semun sem;
        sem.val = 0;semctl(semid, 0, SETVAL, sem);}//写struct sembuf s;
    s.sem_num = 0;
    s.sem_op = 1;
    s.sem_flg = 0;while (1){read(0, p, 256);semop(semid, &s, 1);if (!strcmp(p, "quit\n"))break;}//取消映射shmdt(p);return 0;
}

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <errno.h>
#include <string.h>
#include <unistd.h>union semun {int val;
};int main(int argc, char const *argv[])
{//创建key
    key_t key = ftok(".", 'a');if (key < 0){perror("ftok err");return -1;}printf("key:%d\n", key);//创建或打开共享内存int shmid = shmget(key, 256, IPC_CREAT | IPC_EXCL | 0666);if (shmid < 0){if (errno == 17)
            shmid = shmget(key, 256, 0666);else{perror("shmget err");return -1;}}printf("shmid:%d\n", shmid);//映射char *= shmat(shmid, NULL, 0);if (== (char *)-1){perror("shmat err");return -1;}//创建或打开信号灯集int semid = semget(key, 1, IPC_CREAT | IPC_EXCL | 0666);if (semid <= 0){if (errno == 17)
            semid = semget(key, 1, 0666);else{perror("semget err");return -1;}}else{//初始化union semun sem;
        sem.val = 0;semctl(semid, 0, SETVAL, sem);}//读struct sembuf s;
    s.sem_num = 0;
    s.sem_op = -1;
    s.sem_flg = 0;while (1){semop(semid, &s, 1);if (!strcmp(p, "quit\n"))break;write(1, p, 256);//清空memset(p, 0, 256);}//取消映射shmdt(p);//删除shmctl(shmid, IPC_RMID, NULL);semctl(semid, 0, IPC_RMID);return 0;
}

消息队列

1、特点

消息队列是IPC对象的一种

消息队列由消息队列ID来唯一标识

消息队列就是一个消息的列表。用户可以在消息队列中添加消息、读取消息等。

消息队列可以按照类型来发送(添加)/接收(读取)消息

2.步骤

在不同的进程中,通过相同的key值,拿到相同的消息队列

  1. 创建key值 ftok
  2. 创建或打开消息队列 msgget
  3. 添加消息:按照类型将消息添加到已经打开的消息队列末尾 msgsnd
  4. 读取消息:按照类型把消息从消息队列中读走 msgrcv
  5. 删除消息队列 msgctl

3.函数

1)msgget创建或打开一个消息队列

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgget(key_t key, int flag);
功能:创建或打开一个消息队列
参数:  key值
       flag:创建消息队列的权限IPC_CREAT|IPC_EXCL|0666
返回值:成功:msgid
       失败:-1

2)msgsnd添加消息

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgsnd(int msqid, const void *msgp, size_t size, int flag); 
功能:添加消息
参数:msqid:消息队列的ID
      msgp:指向消息的指针。常用消息结构msgbuf如下:struct msgbuf{long mtype;          //消息类型char mtext[N]};   //消息正文
   size:发送的消息正文的字节数
   flag:IPC_NOWAIT消息没有发送完成函数也会立即返回    0:直到发送完成函数才返回
返回值:成功:0
      失败:-1
使用:msgsnd(msgid, &msg,sizeof(msg)-sizeof(long), 0)
注意:消息结构除了第一个成员必须为long类型外,其他成员可以根据应用的需求自行定义。

3)msgrcv读取消息

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgrcv(int msgid,  void* msgp,  size_t  size,  long msgtype,  int  flag);
功能:读取消息
参数:msgid:消息队列的ID
     msgp:存放读取消息的空间
     size:接受的消息正文的字节数
    msgtype:0:接收消息队列中第一个消息。
            大于0:接收消息队列中第一个类型为msgtyp的消息.
            小于0:接收消息队列中类型值不小于msgtyp的绝对值且类型值又最小的消息。
    flag:0:若无消息函数会一直阻塞
        IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG
返回值:成功:接收到的消息的长度
      失败:-1

  1. msgctl对消息队列的操作,删除消息队列

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgctl ( int msgqid, int cmd, struct msqid_ds *buf );
功能:对消息队列的操作,删除消息队列
参数:msqid:消息队列的队列ID
     cmd:
        IPC_STAT:读取消息队列的属性,并将其保存在buf指向的缓冲区中。
        IPC_SET:设置消息队列的属性。这个值取自buf参数。
        IPC_RMID:从系统中删除消息队列。
     buf:消息队列缓冲区
返回值:成功:0
      失败:-1
用法:msgctl(msgid, IPC_RMID, NULL

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <errno.h>
#include <string.h>struct msgbuf
{long mtype;char name[32];int age;float score;
};int main(int argc, char const *argv[])
{//1、创建key值
    key_t key = ftok(".", 'a');if (key < 0){perror("ftok err");return -1;}printf("key:%d\n", key);//2、创建或打开消息对列int msgid = msgget(key, IPC_CREAT | IPC_EXCL | 0666);if (msgid <= 0){if (errno == EEXIST)
            msgid = msgget(key, 0666);else{perror("msgget err");return -1;}}printf("msgid:%d\n", msgid);//3、添加消息struct msgbuf msg;
    msg.mtype = 1;strcpy(msg.name, "daming");
    msg.age = 18;
    msg.score = 99.99;msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0);    msg.mtype = 3;strcpy(msg.name, "lihua");
    msg.age = 20;
    msg.score = 69.5;msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0);//4、读取消息struct msgbuf msg_r;//0:可以匹配任意类型的消息,拿取得是消息对列得第一条消息msgrcv(msgid, &msg_r, sizeof(msg_r) - sizeof(long),0, 0);printf("name:%s\n", msg_r.name);printf("age:%d\n", msg_r.age);printf("score:%.2f\n", msg_r.score);msgrcv(msgid, &msg_r, sizeof(msg_r) - sizeof(long),3, 0);printf("name:%s\n", msg_r.name);printf("age:%d\n", msg_r.age);printf("score:%.2f\n", msg_r.score);//删除消息队列msgctl(msgid,IPC_RMID,NULL);return 0;
}

4.命令

ipcs -q :查看消息队列

ipcrm -q msgid :删除消息队列

相关文章:

  • 在Linux上编译gdal3.1.2指南
  • 自定义的卷积神经网络模型CNN,对图片进行分类并使用图片进行测试模型-适合入门,从模型到训练再到测试,开源项目
  • 计算机毕业设计选题推荐-超市售货微信小程序/安卓APP-项目实战
  • STM32:I²C通信原理概要
  • 可视化 | 数据可视化降维算法梳理
  • gorilla/websocket的chat示例代码简单分析
  • Web3公链之Cosmos生态的项目Celestia
  • Stable Diffusion系列(一):古早显卡上最新版 WebUI 安装及简单操作
  • Redis Functions 介绍(一)
  • go中“哨兵错误”的由来及使用建议
  • Docker compose容器编排
  • Python 自动化(十六)静态文件处理
  • 在跑腿App系统开发中,如何构建系统架构?
  • 循环神经网络 - RNN
  • MySQL数据库入门到精通——运维篇(1)
  • php的引用
  • [iOS]Core Data浅析一 -- 启用Core Data
  • 「前端早读君006」移动开发必备:那些玩转H5的小技巧
  • 【347天】每日项目总结系列085(2018.01.18)
  • Angular js 常用指令ng-if、ng-class、ng-option、ng-value、ng-click是如何使用的?
  • express.js的介绍及使用
  • IDEA 插件开发入门教程
  • JavaScript设计模式之工厂模式
  • Java新版本的开发已正式进入轨道,版本号18.3
  • macOS 中 shell 创建文件夹及文件并 VS Code 打开
  • MYSQL如何对数据进行自动化升级--以如果某数据表存在并且某字段不存在时则执行更新操作为例...
  • Spark in action on Kubernetes - Playground搭建与架构浅析
  • Spring框架之我见(三)——IOC、AOP
  • vue和cordova项目整合打包,并实现vue调用android的相机的demo
  • 阿里云Kubernetes容器服务上体验Knative
  • 从零到一:用Phaser.js写意地开发小游戏(Chapter 3 - 加载游戏资源)
  • 记一次删除Git记录中的大文件的过程
  • 解析带emoji和链接的聊天系统消息
  • 浏览器缓存机制分析
  • 我从编程教室毕业
  • ​MPV,汽车产品里一个特殊品类的进化过程
  • ​软考-高级-信息系统项目管理师教程 第四版【第19章-配置与变更管理-思维导图】​
  • #Java第九次作业--输入输出流和文件操作
  • (2020)Java后端开发----(面试题和笔试题)
  • (3)nginx 配置(nginx.conf)
  • (C++17) optional的使用
  • (cos^2 X)的定积分,求积分 ∫sin^2(x) dx
  • (pojstep1.1.2)2654(直叙式模拟)
  • (编程语言界的丐帮 C#).NET MD5 HASH 哈希 加密 与JAVA 互通
  • (待修改)PyG安装步骤
  • (论文阅读30/100)Convolutional Pose Machines
  • (新)网络工程师考点串讲与真题详解
  • (已更新)关于Visual Studio 2019安装时VS installer无法下载文件,进度条为0,显示网络有问题的解决办法
  • (转)fock函数详解
  • (转)Mysql的优化设置
  • .NET 指南:抽象化实现的基类
  • .NET/C# 使窗口永不获得焦点
  • @PreAuthorize注解
  • [.net 面向对象程序设计进阶] (19) 异步(Asynchronous) 使用异步创建快速响应和可伸缩性的应用程序...
  • []error LNK2001: unresolved external symbol _m