消息队列、共享内存、信号灯
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的基本用法,帮助新手快速入门并融入这个充满活力的技术社区。 …...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
多场景 OkHttpClient 管理器 - Android 网络通信解决方案
下面是一个完整的 Android 实现,展示如何创建和管理多个 OkHttpClient 实例,分别用于长连接、普通 HTTP 请求和文件下载场景。 <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas…...
【解密LSTM、GRU如何解决传统RNN梯度消失问题】
解密LSTM与GRU:如何让RNN变得更聪明? 在深度学习的世界里,循环神经网络(RNN)以其卓越的序列数据处理能力广泛应用于自然语言处理、时间序列预测等领域。然而,传统RNN存在的一个严重问题——梯度消失&#…...

抖音增长新引擎:品融电商,一站式全案代运营领跑者
抖音增长新引擎:品融电商,一站式全案代运营领跑者 在抖音这个日活超7亿的流量汪洋中,品牌如何破浪前行?自建团队成本高、效果难控;碎片化运营又难成合力——这正是许多企业面临的增长困局。品融电商以「抖音全案代运营…...

2021-03-15 iview一些问题
1.iview 在使用tree组件时,发现没有set类的方法,只有get,那么要改变tree值,只能遍历treeData,递归修改treeData的checked,发现无法更改,原因在于check模式下,子元素的勾选状态跟父节…...

【数据分析】R版IntelliGenes用于生物标志物发现的可解释机器学习
禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍流程步骤1. 输入数据2. 特征选择3. 模型训练4. I-Genes 评分计算5. 输出结果 IntelliGenesR 安装包1. 特征选择2. 模型训练和评估3. I-Genes 评分计…...
Xen Server服务器释放磁盘空间
disk.sh #!/bin/bashcd /run/sr-mount/e54f0646-ae11-0457-b64f-eba4673b824c # 全部虚拟机物理磁盘文件存储 a$(ls -l | awk {print $NF} | cut -d. -f1) # 使用中的虚拟机物理磁盘文件 b$(xe vm-disk-list --multiple | grep uuid | awk {print $NF})printf "%s\n"…...

Python Ovito统计金刚石结构数量
大家好,我是小马老师。 本文介绍python ovito方法统计金刚石结构的方法。 Ovito Identify diamond structure命令可以识别和统计金刚石结构,但是无法直接输出结构的变化情况。 本文使用python调用ovito包的方法,可以持续统计各步的金刚石结构,具体代码如下: from ovito…...
怎么让Comfyui导出的图像不包含工作流信息,
为了数据安全,让Comfyui导出的图像不包含工作流信息,导出的图像就不会拖到comfyui中加载出来工作流。 ComfyUI的目录下node.py 直接移除 pnginfo(推荐) 在 save_images 方法中,删除或注释掉所有与 metadata …...
6个月Python学习计划 Day 16 - 面向对象编程(OOP)基础
第三周 Day 3 🎯 今日目标 理解类(class)和对象(object)的关系学会定义类的属性、方法和构造函数(init)掌握对象的创建与使用初识封装、继承和多态的基本概念(预告) &a…...