消息队列、共享内存、信号灯
IPC(Inter-Process Communication,进程间通信)
常用的 IPC 对象包括管道(pipe)、消息队列(message queue)、信号量(semaphore)和共享内存(shared memory)等
1.ipcs
查看系统重的消息队列、共享内存、信号灯的信息
2.ipcrm
删除消息队列、共享内存、信号灯
ipcrm -Q/-M/-S key
ipcrm -q/-m/-s 消息队列ID/共享内存ID/信号灯ID
3.操作流程:
创建消息队列 -> 发送消息 -> 接收消息
4.函数接口:
1.ftok
key_t ftok(const char *pathname, int proj_id);
功能:
根据pathname和proj_id生成一个key_t类型的key值,将来可以用来创建消息队列、共享内存、信号灯
参数:
pathname:文件路径
proj_id:8位非0值
返回值:
成功返回key_t类型的IPC对象的key值
失败返回-1
2.msgget
int msgget(key_t key, int msgflg);
功能:
根据key值对象的IPC对象创建一个消息队列
参数:
key:IPC对象名字
msgflg:IPC_CREAT 对象不存在就创建
IPC_EXCL 对象存在报错
IPC_CREAT | 0664
返回值:
成功返回消息队列ID
失败返回-1
3.msgsnd
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
功能:
向消息队列中发送消息
参数:
msqid:消息队列的ID号
msgp:发送消息空间的首地址
struct msgbuf {
long mtype; /* message type, must be > 0 */ 消息的类型号
char mtext[1]; /* message data */ 消息的内容
};
msgz:发送消息内容的大小(不包含发送消息类型)
msgflg:属性,默认为0
返回值:
成功返回0
失败返回-1
4.msgrcv
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);
功能:
从消息队列中接收消息
参数:
msqid:消息队列的ID号
msgp:存放接收到消息空间的首地址
msgsz:最多接收消息的空间的大小
msgtyp:想要接收消息的类型
msgflg:属性,默认为0
返回值:
成功返回实际接收的字节数
失败返回-1
5.msgctl
int msgctl(int msqid, int cmd, struct msqid_ds *buf);
功能:
向消息队列发送一条cmd命令
参数:
msqid:消息队列的ID号
cmd:IPC_RMID 删除消息队列
buf:默认传NULL
返回值:
成功返回0
失败返回-1
练习: 利用消息队列实现clientA和clientB两个进程任务的全双工聊天功能
#ifndef __HEAD_H__
#define __HEAD_H__#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>#include <pthread.h>
#include <semaphore.h>
#include <signal.h>
#include <sys/ipc.h>
#include <sys/msg.h>struct msgbuf
{long mtype;char mtext[256];
};#endif
Client A
#include "head.h"pthread_t tid_send;
pthread_t tid_recv;
int msgid = 0;void *SendThread(void *arg)
{struct msgbuf sendmsg;int ret = 0;while (1){memset(&sendmsg, 0, sizeof(sendmsg));sendmsg.mtype = 100;gets(sendmsg.mtext);ret = msgsnd(msgid, &sendmsg, sizeof(sendmsg) - sizeof(long), 0);if (-1 == ret){perror("fail to msgsnd");return NULL;}if (!strcmp(sendmsg.mtext, ".quit")){break;}}pthread_cancel(tid_recv);return NULL;
}void *RecvThread(void *arg)
{struct msgbuf recvmsg;ssize_t nsize = 0;while (1){memset(&recvmsg, 0, sizeof(recvmsg));nsize = msgrcv(msgid, &recvmsg, sizeof(recvmsg) - sizeof(long), 200, 0);if (-1 == nsize){perror("fail to msgrcv");return NULL;}if (!strcmp(recvmsg.mtext, ".quit")){break;}printf("RECV:%s\n", recvmsg.mtext);}pthread_cancel(tid_send);return NULL;
}int main(void)
{key_t key;key = ftok(".", 'a');if (-1 == key){perror("fail to ftok");return -1;}msgid = msgget(key, IPC_CREAT | 0664);if (-1 == msgid){perror("fail to msgget");return -1;}pthread_create(&tid_send, NULL, SendThread, NULL);pthread_create(&tid_recv, NULL, RecvThread, NULL);pthread_join(tid_send, NULL);pthread_join(tid_recv, NULL);msgctl(msgid, IPC_RMID, NULL);return 0;
}
clientB
#include "head.h"pthread_t tid_send;
pthread_t tid_recv;
int msgid = 0;void *SendThread(void *arg)
{struct msgbuf sendmsg;int ret = 0;while (1){memset(&sendmsg, 0, sizeof(sendmsg));sendmsg.mtype = 200;gets(sendmsg.mtext);ret = msgsnd(msgid, &sendmsg, sizeof(sendmsg) - sizeof(long), 0);if (-1 == ret){perror("fail to msgsnd");return NULL;}if (!strcmp(sendmsg.mtext, ".quit")){break;}}pthread_cancel(tid_recv);return NULL;
}void *RecvThread(void *arg)
{struct msgbuf recvmsg;ssize_t nsize = 0;while (1){memset(&recvmsg, 0, sizeof(recvmsg));nsize = msgrcv(msgid, &recvmsg, sizeof(recvmsg) - sizeof(long), 100, 0);if (-1 == nsize){perror("fail to msgrcv");return NULL;}if (!strcmp(recvmsg.mtext, ".quit")){break;}printf("RECV:%s\n", recvmsg.mtext);}pthread_cancel(tid_send);return NULL;
}int main(void)
{key_t key;key = ftok(".", 'a');if (-1 == key){perror("fail to ftok");return -1;}msgid = msgget(key, IPC_CREAT | 0664);if (-1 == msgid){perror("fail to msgget");return -1;}pthread_create(&tid_send, NULL, SendThread, NULL);pthread_create(&tid_recv, NULL, RecvThread, NULL);pthread_join(tid_send, NULL);pthread_join(tid_recv, NULL);msgctl(msgid, IPC_RMID, NULL);return 0;
}
2.共享内存:
它是进程间通信最高效的形式
1.操作方式:
创建共享内存 -> 映射到共享内存中 -> 共享内存操作 -> 解除映射 -> 删除共享内存
2.函数接口:
1.ftok
2.shmget
int shmget(key_t key, size_t size, int shmflg);
功能:
创建一个共享内存
参数:
key:IPC对象名称
size:共享内存的大小
shmflg:
IPC_CREAT
IPC_EXCL
返回值:
成功返回共享内存ID
失败返回-1
3.shmat
void *shmat(int shmid, const void *shmaddr, int shmflg);
功能:
将一个地址映射到共享内存中
参数:
shmid:共享内存ID号
shmaddr:NULL 让系统选择一个合适的地址映射
不为NULL shmflg 设定为SHM_RND 选择离给定地址最近的能够映射的地址进行映射
否则传递地址为4k的整数倍
返回值:
成功返回映射到共享内存空间中的地址
失败返回NULL
4.shmdt
int shmdt(const void *shmaddr);
功能:
解除映射
参数:
shmaddr:映射的地址
返回值:
成功返回0
失败返回-1
5.shmctl
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
功能:
向共享内存发送命令
参数:
shmid:共享内存ID号
cmd:IPC_RMID 删除共享内存
buf:NULL
返回值:
成功返回0
失败返回-1
3.信号灯(有名信号量)
1.创建
semget
int semget(key_t key, int nsems, int semflg);
功能:
创建一组信号量
参数:
key:IPC对象名
nsems:信号量的个数
semflg:IPC_CREAT
返回值:
成功返回信号量ID
失败返回-1
2.销毁
semctl
int semctl(int semid, int semnum, int cmd, ...);
功能:
向信号灯发送命令
参数:
semid:信号灯ID号
semnum:具体操作信号量的编号
cmd:
IPC_RMID 删除信号灯
SETVAL 设置信号量的值
返回值:
成功返回0
失败返回-1初始化: //注意:这是一个共用体
union semun {
int val; /* Value for SETVAL */
struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */
unsigned short *array; /* Array for GETALL, SETALL */
struct seminfo *__buf; /* Buffer for IPC_INFO
(Linux-specific) */
};
3.申请信号量
4.释放信号量
semop
int semop(int semid, struct sembuf *sops, size_t nsops);
功能:
对信号量完成操作
参数:
semid:信号灯的ID号
sops:信号量操作的数组首地址
nsops:数组元素个数
返回值:
成功返回0
失败返回-1unsigned short sem_num; /* semaphore number */ 操作信号量的下标
short sem_op; /* semaphore operation */ 具体对信号量的操作(申请:-1 释放:+1)
short sem_flg; /* operation flags */ SEM_UNDO
#include "head.h"union semun {int val; /* Value for SETVAL */struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */unsigned short *array; /* Array for GETALL, SETALL */struct seminfo *__buf; /* Buffer for IPC_INFO(Linux-specific) */
};int main(void)
{key_t key;int semid = 0;union semun myun;struct sembuf mybuf;int ret = 0;key = ftok(".", 'a');if (-1 == key){perror("fail to ftok");return -1;}semid = semget(key, 2, IPC_CREAT | 0664);if (-1 == semid){perror("fail to semget");return -1;}/* 对信号灯中的0号信号量初始化为0 */myun.val = 0;semctl(semid, 0, SETVAL, myun);/* 对信号灯中的1号信号量初始化为1 */myun.val = 1;semctl(semid, 1, SETVAL, myun);/* 申请1号信号量 */mybuf.sem_num = 1;mybuf.sem_op = -1;mybuf.sem_flg = SEM_UNDO;ret = semop(semid, &mybuf, 1);if (-1 == ret){perror("fail to semop");return -1;}printf("申请到写信号量!\n");/* 释放0号信号量 */mybuf.sem_num = 0;mybuf.sem_op = +1;mybuf.sem_flg = SEM_UNDO;ret = semop(semid, &mybuf, 1);if (-1 == ret){perror("fail to semop");return -1;}printf("释放了读信号量!\n");/* 申请0号信号量 */mybuf.sem_num = 0;mybuf.sem_op = -1;mybuf.sem_flg = SEM_UNDO;ret = semop(semid, &mybuf, 1);if (-1 == ret){perror("fail to semop");return -1;}printf("申请了读信号量!\n");semctl(semid, 0, IPC_RMID);return 0;
}
练习:使用共享内存和信号量实现同步通信
#ifndef __HEAD_H__
#define __HEAD_H__#include <sys/types.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <dirent.h>
#include <time.h>
#include <pwd.h>
#include <grp.h>
#include <pthread.h>
#include <semaphore.h>
#include <signal.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/shm.h>
#include <sys/sem.h>union semun {int val; /* Value for SETVAL */struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */unsigned short *array; /* Array for GETALL, SETALL */struct seminfo *__buf; /* Buffer for IPC_INFO(Linux-specific) */
};extern int init_sem(int semid, int *parray, int len);
extern int sem_p(int semid, int num);
extern int sem_v(int semid, int num);#endif
发送
#include "head.h"int main(void)
{key_t key;int shmid = 0;int semid = 0;char *pshmaddr = NULL;int val[2] = {0, 1};key = ftok(".", 'a');if (-1 == key){perror("fail to ftok");return -1;}semid = semget(key, 2, IPC_CREAT | 0664);if (-1 == semid){perror("fail to semget");return -1;}init_sem(semid, val, 2);shmid = shmget(key, 4096, IPC_CREAT | 0664);if (-1 == shmid){perror("fail to shmget");return -1;}pshmaddr = shmat(shmid, NULL, 0);if (NULL == pshmaddr){perror("fail to shmat");return -1;}while (1){sem_p(semid, 1);gets(pshmaddr);sem_v(semid, 0);if (!strcmp(pshmaddr, ".quit")){break;}}shmdt(pshmaddr);shmctl(shmid, IPC_RMID, NULL);return 0;
}
接收
#include "head.h"int main(void)
{key_t key;int shmid = 0;int semid = 0;char *pshmaddr = NULL;int val[2] = {0, 1};key = ftok(".", 'a');if (-1 == key){perror("fail to ftok");return -1;}semid = semget(key, 2, IPC_CREAT | 0664);if (-1 == semid){perror("fail to semget");return -1;}init_sem(semid, val, 2);shmid = shmget(key, 4096, IPC_CREAT | 0664);if (-1 == shmid){perror("fail to shmget");return -1;}pshmaddr = shmat(shmid, NULL, 0);if (NULL == pshmaddr){perror("fail to shmat");return -1;}while (1){sem_p(semid, 0);printf("SHMADDR:%s\n", pshmaddr);if (!strcmp(pshmaddr, ".quit")){break;}sem_v(semid, 1);}shmdt(pshmaddr);shmctl(shmid, IPC_RMID, NULL);return 0;
}
信号量控制
#include "head.h"int init_sem(int semid, int *parray, int len)
{union semun myun;int i = 0;int ret = 0;for (i = 0; i < len; i++){myun.val = parray[i];ret = semctl(semid, i, SETVAL, myun);if (-1 == ret){perror("fail to semctl");return -1;}}return 0;
}int sem_p(int semid, int num)
{int ret = 0;struct sembuf mybuf;mybuf.sem_num = num;mybuf.sem_op = -1;mybuf.sem_flg = SEM_UNDO;ret = semop(semid, &mybuf, 1);if (-1 == ret){perror("fail to semop");return -1;}return 0;
}int sem_v(int semid, int num)
{int ret = 0;struct sembuf mybuf;mybuf.sem_num = num;mybuf.sem_op = +1;mybuf.sem_flg = SEM_UNDO;ret = semop(semid, &mybuf, 1);if (-1 == ret){perror("fail to semop");return -1;}return 0;
}
相关文章:
消息队列、共享内存、信号灯
IPC(Inter-Process Communication,进程间通信) 常用的 IPC 对象包括管道(pipe)、消息队列(message queue)、信号量(semaphore)和共享内存(shared memory&…...
K次取反后最大化的数组和 加油站 分发糖果 柠檬水找零
1005.K次取反后最大化的数组和 力扣题目链接(opens new window) 给定一个整数数组 A,我们只能用以下方法修改该数组:我们选择某个索引 i 并将 A[i] 替换为 -A[i],然后总共重复这个过程 K 次。(我们可以多次选择同一个索引 i。&a…...
Standoff: 独特的基于真实商业基础架构的网络战
Standoff 网络战通常每年进行两次(5 月和 11 月)。该公共活动的核心是由多个良心黑客团队(也称为威胁研究团队、白帽队或红队)对虚拟地区的基础架构进行系列攻击。 下一届 Standoff 将在 2024 年 5 月 23 日至 26 日举行的 Posi…...
如何成为fpga工程师
FPGA的应用领域非常的广,尤其再人工智能,大数据,云计算等等方向非常吃香。加上国家这两年的政策支持,整个芯片行业相比较其他的传统行业来说会好很多,总之前景是光明的,道路是曲折的,想要在人才…...
基础算法(二)#蓝桥杯
文章目录 8、双指针8.1、挑选子串8.2、聪明的小羊肖恩8.3、神奇的数组 9、二分9.1、跳石头9.2、可凑成的最大花朵数9.3、最大通过数9.4、妮妮的月饼广场9.5、基德的神秘冒险9.6、体育健将 10、倍增10.1、快速幂10.2、最近公共祖先LCA查询10.3、理想之城10.4、数的变换 8、双指针…...
运筹学_1.1.4 线性规划问题-解的概念
1.1.4 线性规划问题-解的概念 一、可行解与最优解二、基的概念三、基变量、基向量;非基变量、非基向量;基解、基可行解;四、最优解与可行解、基可行解的关系五、用例题(枚举法)巩固基解、基可行解、最优解三个概念1、例…...
物联网主机:为智能交通赋能
物联网(IoT)技术的发展为智能交通领域带来了许多创新的解决方案。而在物联网应用中,物联网主机起着关键的作用。本文将为大家介绍一款名为E6000的物联网主机,它是一种多协议、多接口的物联网主机,为智能交通系统的建设…...
「Vue3系列」Vue3简介及安装
文章目录 一、Vue3简介二、Vue3安装三、Vue3应用案例四、package.json详解五、相关链接 一、Vue3简介 Vue3是Vue.js框架的第三个主要版本,于2020年9月18日发布,代号为“One Piece”。Vue3在性能、体积、TypeScript支持、API设计等方面都有显著的提升和改…...
Javascript:分支语句
一、前言 关于分支语句的介绍来啦,开始记笔记。 二、正文 1.分支语句if 分支语句就是通过判断已给的表达式的条件来执行语句,表达式为真才能执行. if(条件){满足条件才要执行的代码} 条件为true时,才能进行大括号的代码。 除了空字符串&am…...
从零开始学习PX4源码2(PX4姿态误差计算)
目录 文章目录 目录摘要1.源码1.1源码路径1.2源码程序1.3源码功能 2.源码分析 摘要 本节主要记录PX4姿态误差计算过程,欢迎批评指正。 1.源码 1.1源码路径 PX4-Autopilot/src/modules/mc_att_control/AttitudeControl/AttitudeControl.cpp1.2源码程序 matrix::…...
git安装与使用4.3
一、git的安装 1、下载git包 下载git包url:https://git-scm.com/download/win 下载包分为:64位和32位 2、点击安装包 2、选择安装路径 3、 点击下一步 4、点击next 5、点击next 6、点击next 7、 8、 9、 10、 11、 12、在桌面空白处,右键…...
Python:关于数据服务中的Web API的设计
搭建类似joinquant、tushare类似的私有数据服务应用,有以下一些点需要注意: 需要说明的是,这里讨论的是web api前后端,当然还有其它方案,thrift,grpc等。因为要考虑到一鱼两吃,本文只探讨web ap…...
VMwareWorkstation17.0虚拟机安装搭建PcDos2000虚拟机(完整图文详细步骤教程)
VMwareWorkstation17.0虚拟机安装搭建PcDos2000虚拟机(完整图文详细步骤教程) 一、PcDos20001.PcDos2000简介2.PcDos2000下载 二、创建PcDos2000虚拟机1.新建虚拟机2.类型配置3.类型配置4.选择版本5.命名、存位置6.磁盘容量7.调整虚拟配置7.1 调整虚拟配…...
第七个程序:两个字符串连接后计算长度
实验步骤; 第一步:新建项目 第二步:程序编写 第三步:运行结果 Labview一共7个字节,长度为7,一个字母一个字节 汉字为2个字节,图一为4,图二为8 所以结果分别为11和15 视频教学: 字…...
【大数据】-- dataworks 创建odps 的 hudi 外表
文档:创建OSS外部表_云原生大数据计算服务 MaxCompute(MaxCompute)-阿里云帮助中心 举例:创建 odps 的 hudi 外表 CREATE EXTERNAL TABLE IF NOT EXISTS my_project.ods_hudi_mysql_words_h_all (id BIGINT COMMENT 主键id,`words` STRING COMMENT 词…...
ChatGPT与GEE+ENVI+python高光谱,多光谱等成像遥感数据处理技术
原文链接:ChatGPT与GEEENVIpython高光谱,多光谱等成像遥感技术 第一遥感科学与AI基础 一:遥感科学的基本原理和历史 从摄影侦察到卫星图像 遥感的基本原理 遥感的典型应用 最新进展和未来趋势 二:ChatGPT 什么是ChatGPT&a…...
学习linux从0到初级工程师-3
一、LNMP 1.1 搭建LNMP LNMP:LinuxNginxMysqlPHP LNMP优势: 1.web服务器一种,Nginx处理静态文件、索引文件,自动索引的效率非常高; 2.作为代理服务器,Nginx可以实现无缓存的反向代理加速,提高网站运行…...
java实现文件上传到本地
很多时候我们都需要进行文件上传和下载的操作,具体怎么实现网上的代码其实也是挺多的,刚好我的项目中也遇到了文件上传和下载的需求,本篇博文具体讲解上传操作,下篇博文讲解下载操作。 我们具体来想一想要将一个从前端传来的文件…...
基于springboot+vue的多媒体素材库的开发与应用系统
博主主页:猫头鹰源码 博主简介:Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战,欢迎高校老师\讲师\同行交流合作 主要内容:毕业设计(Javaweb项目|小程序|Pyt…...
《GitHub新手入门指南:从零开始掌握基本用法》
在现代软件开发和技术社区中,GitHub已经成为了一个不可或缺的平台。它不仅是一个代码托管平台,更是一个技术交流、学习分享的社交平台。但对于初学者来说,GitHub可能会有些令人望而却步。本文将详细介绍GitHub的基本用法,帮助新手快速入门并融入这个充满活力的技术社区。 …...
java_网络服务相关_gateway_nacos_feign区别联系
1. spring-cloud-starter-gateway 作用:作为微服务架构的网关,统一入口,处理所有外部请求。 核心能力: 路由转发(基于路径、服务名等)过滤器(鉴权、限流、日志、Header 处理)支持负…...
AI Agent与Agentic AI:原理、应用、挑战与未来展望
文章目录 一、引言二、AI Agent与Agentic AI的兴起2.1 技术契机与生态成熟2.2 Agent的定义与特征2.3 Agent的发展历程 三、AI Agent的核心技术栈解密3.1 感知模块代码示例:使用Python和OpenCV进行图像识别 3.2 认知与决策模块代码示例:使用OpenAI GPT-3进…...
关于nvm与node.js
1 安装nvm 安装过程中手动修改 nvm的安装路径, 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解,但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后,通常在该文件中会出现以下配置&…...
生成 Git SSH 证书
🔑 1. 生成 SSH 密钥对 在终端(Windows 使用 Git Bash,Mac/Linux 使用 Terminal)执行命令: ssh-keygen -t rsa -b 4096 -C "your_emailexample.com" 参数说明: -t rsa&#x…...
spring:实例工厂方法获取bean
spring处理使用静态工厂方法获取bean实例,也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下: 定义实例工厂类(Java代码),定义实例工厂(xml),定义调用实例工厂ÿ…...
Robots.txt 文件
什么是robots.txt? robots.txt 是一个位于网站根目录下的文本文件(如:https://example.com/robots.txt),它用于指导网络爬虫(如搜索引擎的蜘蛛程序)如何抓取该网站的内容。这个文件遵循 Robots…...
3403. 从盒子中找出字典序最大的字符串 I
3403. 从盒子中找出字典序最大的字符串 I 题目链接:3403. 从盒子中找出字典序最大的字符串 I 代码如下: class Solution { public:string answerString(string word, int numFriends) {if (numFriends 1) {return word;}string res;for (int i 0;i &…...
rnn判断string中第一次出现a的下标
# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...
SiFli 52把Imagie图片,Font字体资源放在指定位置,编译成指定img.bin和font.bin的问题
分区配置 (ptab.json) img 属性介绍: img 属性指定分区存放的 image 名称,指定的 image 名称必须是当前工程生成的 binary 。 如果 binary 有多个文件,则以 proj_name:binary_name 格式指定文件名, proj_name 为工程 名&…...
宇树科技,改名了!
提到国内具身智能和机器人领域的代表企业,那宇树科技(Unitree)必须名列其榜。 最近,宇树科技的一项新变动消息在业界引发了不少关注和讨论,即: 宇树向其合作伙伴发布了一封公司名称变更函称,因…...
