当前位置: 首页 > 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 :删除消息队列

相关文章:

信号灯集,消息队列

信号灯集 1、概念 信号灯(semaphore)&#xff0c;也叫信号量。它是不同进程间或一个给定进程内部不同线程间同步的机制&#xff1b;System V的信号灯是一个或者多个信号灯的一个集合。其中的每一个都是单独的计数信号灯。而Posix信号灯指的是单个计数信号灯。 通过信号灯集实现…...

我在Vscode学OpenCV 初步接触

OpenCV是一个开源的计算机视觉库&#xff0c;可以处理图像和视频数据。它包含了超过2500个优化过的算法&#xff0c;用于对图像和视频进行处理&#xff0c;包括目标识别、面部识别、运动跟踪、立体视觉等。OpenCV支持多种编程语言&#xff0c;包括C、Python、Java等&#xff0c…...

[threejs]让导入的gltf模型显示边框

边框1效果图如下&#xff1a; 代码如下&#xff1a; const gltfLoader1 new GLTFLoader();gltfLoader1.load( "/assets/box/1/scene.gltf" ,function(gltf){let model gltf.scene;model.scale.set(3,3,3)// scene1.add(model);// renderer1.render(scene1, camera…...

YOLOv5优化:独家创新(SC_C_Detect)检测头结构创新,实现涨点 | 检测头新颖创新系列

💡💡💡本文独家改进:独家创新(SC_C_Detect)检测头结构创新,适合科研创新度十足,强烈推荐 SC_C_Detect | 亲测在多个数据集能够实现大幅涨点 目录 1. SC_C_Detect介绍 2. SC_C_Detect加入YOLOv5 2.1 新建models/head_improve.py...

作物模型--土壤数据制备过程

作物模型–土壤数据制备过程 首先打开FAO网站 下载下面这两个 Arcgis打开.bil文件 .mdb文件在access中转成.xls格式 Arcgis中对.bil文件定义投影...

学习笔记|单样本t检验|无统计学意义|规范表达|《小白爱上SPSS》课程:SPSS第四讲 | 单样本T检验怎么做?很单纯很简单!

目录 学习目的软件版本原始文档一、实战案例二、案例解析本案例之目的 四、SPSS操作1、正态性检验Tips&#xff1a;无统计学意义 2、t检验结果 五、结果解读六、规范报告1、规范表格2、规范文字 注意划重点 学习目的 SPSS第四讲 | 单样本T检验怎么做&#xff1f;很单纯很简单&…...

Bug管理规范

1BUG定义 1.1Bug状态 BUG状态标记BUG当前所处的状态&#xff0c;是用来处理BUG流程的主要参数&#xff0c;JIRA缺陷管理平台有以下一些状态&#xff1a; 新增&#xff08;New&#xff09;&#xff1a;测试人员新发现的系统Bug&#xff1b; 打开&#xff08;Open&#xf…...

剑指JUC原理-8.Java内存模型

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring源码、JUC源码&#x1f525;如果感觉博主的文章还不错的话&#xff0c;请&#x1f44d;三连支持&…...

Azure 机器学习 - 使用 AutoML 和 Python 训练物体检测模型

目录 一、Azure环境准备二、计算目标设置三、试验设置四、直观呈现输入数据五、上传数据并创建 MLTable六、配置物体检测试验适用于图像任务的自动超参数扫描 (AutoMode)适用于图像任务的手动超参数扫描作业限制 七、注册和部署模型获取最佳试用版注册模型配置联机终结点创建终…...

【深度学习】pytorch——快速入门

笔记为自我总结整理的学习笔记&#xff0c;若有错误欢迎指出哟~ pytorch快速入门 简介张量&#xff08;Tensor&#xff09;操作创建张量向量拷贝张量维度张量加法函数名后面带下划线 _ 的函数索引和切片Tensor和Numpy的数组之间的转换张量&#xff08;tensor&#xff09;与标量…...

git本地项目同时推送提交到github和gitee同步

git本地项目同时推送提交到github和gitee同步 同时推送到GitHub和Gitee&#xff08;码云&#xff09;可以通过设置多个远程仓库地址来实现。具体步骤如下&#xff1a; 一、分别推送 # 初始化仓库 git init# 添加远程仓库 git remote add gitee gitgitee.com:bealei/test.git…...

结构体数据类型使用的一些注意点

1.结构体定义时的注意事项&#xff1a; 1.错误定义结构体&#xff1a; struct students {char name[9] "Mike";int height 185; }; 这是不对的&#xff0c;在 C 语言中&#xff0c;这是由语言的设计原则所决定的。结构体的定义&#xff08;struct declaration&…...

Serverless化云产品超40款 阿里云发布全球首款容器计算服务

10月31日&#xff0c;杭州云栖大会上&#xff0c;阿里云宣布推出全球首款容器计算服务ACS&#xff0c;大幅提升操作的易用性并节省20%资源成本&#xff0c;真正将Serverless理念大规模落地&#xff0c;同时阿里云 Serverless化进程进入快车道&#xff0c;有超40款云产品提供了S…...

最小化安装移动云大云操作系统--BCLinux-R8-U2-Server-x86_64-231017版

有个业务系统因为兼容性问题&#xff0c;需要安装el8.2的系统&#xff0c;因此对应安装国产环境下的BCLinuxR8U2系统来满足用户需求。BCLinux-R8-U2-Server是中国移动基于AnolisOS8.2深度定制的企业级X86服务器通用版操作系统。本文记录在DELL PowerEdge R720xd服务器上最小化安…...

索引创建的原则

索引的创建是数据库优化中非常重要的一部分&#xff0c;正确创建索引可以大大提高查询效率。以下是一些创建索引时需要考虑的原则&#xff1a; 根据查询频率创建索引&#xff1a; 频繁用于检索的列&#xff1a; 那些频繁用于查询的列或经常出现在 WHERE、JOIN、ORDER BY 和 GR…...

动态表单生成Demo(Vue+elment)

摘要&#xff1a;本文将介绍如何使用vue和elment ui组件库实现一个简单的动态表单生成的Demo。主要涉及两个.vue文件的书写&#xff0c;一个是动态表单生成的组件文件&#xff0c;一个是使用该动态表单生成的组件。 1.动态表单生成组件 这里仅集成了输入框、选择框、日期框三种…...

JMeter断言之JSON断言

JSON断言 若服务器返回的Response Body为JSON格式的数据&#xff0c;使用JSON断言来判断测试结果是较好的选择。 首先需要根据JSON Path从返回的JSON数据中提取需要判断的实际结果&#xff0c;再设置预期结果&#xff0c;两者进行比较得出断言结果。 下面首先介绍JSON与JSON…...

LuatOS-SOC接口文档(air780E)--mqtt - mqtt客户端

常量 常量 类型 解释 mqtt.STATE_DISCONNECT number mqtt 断开 mqtt.STATE_SCONNECT number mqtt socket连接中 mqtt.STATE_MQTT number mqtt socket已连接 mqtt连接中 mqtt.STATE_READY number mqtt mqtt已连接 mqttc:subscribe(topic, qos) 订阅主题 参数 …...

安装Python环境

Python 安装包下载地址&#xff1a;https://www.python.org/downloads/ 打开该链接&#xff0c;可以看到有两个版本的 Python&#xff0c;分别是 Python 3.x 和 Python 2.x&#xff0c;如下图所示&#xff1a; Python下载页面截图 图 1 Python 下载页面截图&#xff08;包含…...

[nodejs] 爬虫加入并发限制并发实现痞客邦网页截图

今晚想给偶像的相册截个图,避免某一天网站挂了我想看看回忆都不行,用的是js的木偶师来爬虫台湾的部落格,效果图大概是这样,很不错 问题来了.我很贪心, 我想一次性把相册全爬了,也就是并发 ,这个人的相册有19个!!我一下子要开19个谷歌浏览器那个什么进程, 然后程序就崩了, 我就想…...

linux 错误码总结

1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...

Module Federation 和 Native Federation 的比较

前言 Module Federation 是 Webpack 5 引入的微前端架构方案&#xff0c;允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...

【C语言练习】080. 使用C语言实现简单的数据库操作

080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...

06 Deep learning神经网络编程基础 激活函数 --吴恩达

深度学习激活函数详解 一、核心作用 引入非线性:使神经网络可学习复杂模式控制输出范围:如Sigmoid将输出限制在(0,1)梯度传递:影响反向传播的稳定性二、常见类型及数学表达 Sigmoid σ ( x ) = 1 1 +...

3-11单元格区域边界定位(End属性)学习笔记

返回一个Range 对象&#xff0c;只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意&#xff1a;它移动的位置必须是相连的有内容的单元格…...

智能AI电话机器人系统的识别能力现状与发展水平

一、引言 随着人工智能技术的飞速发展&#xff0c;AI电话机器人系统已经从简单的自动应答工具演变为具备复杂交互能力的智能助手。这类系统结合了语音识别、自然语言处理、情感计算和机器学习等多项前沿技术&#xff0c;在客户服务、营销推广、信息查询等领域发挥着越来越重要…...

Python Ovito统计金刚石结构数量

大家好,我是小马老师。 本文介绍python ovito方法统计金刚石结构的方法。 Ovito Identify diamond structure命令可以识别和统计金刚石结构,但是无法直接输出结构的变化情况。 本文使用python调用ovito包的方法,可以持续统计各步的金刚石结构,具体代码如下: from ovito…...

Caliper 配置文件解析:fisco-bcos.json

config.yaml 文件 config.yaml 是 Caliper 的主配置文件,通常包含以下内容: test:name: fisco-bcos-test # 测试名称description: Performance test of FISCO-BCOS # 测试描述workers:type: local # 工作进程类型number: 5 # 工作进程数量monitor:type: - docker- pro…...

在树莓派上添加音频输入设备的几种方法

在树莓派上添加音频输入设备可以通过以下步骤完成&#xff0c;具体方法取决于设备类型&#xff08;如USB麦克风、3.5mm接口麦克风或HDMI音频输入&#xff09;。以下是详细指南&#xff1a; 1. 连接音频输入设备 USB麦克风/声卡&#xff1a;直接插入树莓派的USB接口。3.5mm麦克…...

2025年- H71-Lc179--39.组合总和(回溯,组合)--Java版

1.题目描述 2.思路 当前的元素可以重复使用。 &#xff08;1&#xff09;确定回溯算法函数的参数和返回值&#xff08;一般是void类型&#xff09; &#xff08;2&#xff09;因为是用递归实现的&#xff0c;所以我们要确定终止条件 &#xff08;3&#xff09;单层搜索逻辑 二…...