笔记六:进程间的通信(IPC通信之消息队列)
。消息被发送到队列中。“消息队列”是在消息的传输过程中保存消息的容器。消息队列管理器在将消息从它的源中继到它的目标时充当中间人。队列的主要目的是提供路由并保证消息的传递;如果发送消息时接收者不可用,消息队列会保留消息,直到可以成功地传递它。
sqid_ds 维护消息队列的结构体,队列的第一个消息指针msg_first,最后一个消息指针msg_last
|
文件I/O
|
消息队列
|
|
open
|
msg_get
|
|
read
|
msgrcv
|
|
write
|
msgsnd
|
|
close
|
msgctrl
|
|
所需头文件
|
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
|
|
函数原型
|
int msgget(key_t key, int flag);
|
|
函数参数
|
key:和消息队列关联的key值
|
|
flag:消息队列的访问权限
|
|
|
函数返回值
|
成功:消息队列ID
|
|
出错:-1
|
|
所需头文件
|
#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
|
||
#include "sys/ipc.h"
#include "sys/msg.h"
#include "sys/types.h"
#include "stdio.h"
struct mybuf
{
long mytype;
char sendbuf[128];
};
int main()
{
int key;
int msgid;
struct mybuf wrbuf,rdbuf;
key=ftok("./a.c",'a');
if(key < 0)
{
printf("creat key failure\n");
return -1;
}
msgid=msgget(key,IPC_CREAT | 0777);
if(msgid < 0)
{
printf("creat message queue failure\n");
return -2;
}
wrbuf.mytype=10;// voltage data
printf("please input data:\n");
fgets(wrbuf.sendbuf,128,stdin);
msgsnd(msgid,(void *)&wrbuf,128,0);
system("ipcs -q");
//read
msgrcv(msgid,(void *)&rdbuf,128,11,0);
printf("%s",rdbuf.sendbuf);
system("ipcs -q");
msgctl(msgid,IPC_RMID,NULL);
return 0;
}
|
所需头文件
|
#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);
|
|
|
函数参数
|
msqid:消息队列的ID
|
|
|
msgp:接收消息的缓冲区
|
||
|
size:要接收的消息的字节数
|
||
|
msgtype:
|
0:接收消息队列中第一个消息。
|
|
|
大于0:接收消息队列中第一个类型为msgtyp的消息.
|
||
|
小于0:接收消息队列中类型值不大于msgtyp的绝对值且类型值又最小的消息。
|
||
|
flag:
|
0:若无消息函数会一直阻塞
|
|
|
IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG。
|
||
|
函数返回值
|
成功:接收到的消息的长度
|
|
|
出错:-1
|
||
|
所需头文件
|
#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
|
|
server.c 端
#include "sys/ipc.h"
#include "sys/msg.h"
#include "sys/types.h"
#include "stdio.h"
#include "string.h"
struct mybuf
{
long mytype;
char sendbuf[128];
};
int main()
{
int key;
int msgid;
int pid;
struct mybuf wrbuf,rdbuf;
key=ftok("./a.c",'a');
if(key < 0)
{
printf("creat key failure\n");
return -1;
}
msgid=msgget(key,IPC_CREAT | 0777);
if(msgid < 0)
{
printf("creat message queue failure\n");
return -2;
}
pid=fork();
if(pid < 0)
return -3;
if(pid > 0)//write type=10
{
wrbuf.mytype=10;// voltage data
while(1)
{
memset(wrbuf.sendbuf,0,128);
printf("please input data:\n");
fgets(wrbuf.sendbuf,128,stdin);
msgsnd(msgid,(void *)&wrbuf,128,0);
}
}
//read
if(pid == 0) //read type=11
{
while(1)
{
memset(rdbuf.sendbuf,0,128);
msgrcv(msgid,(void *)&rdbuf,128,11,0);
printf("%s",rdbuf.sendbuf);
}
}
msgctl(msgid,IPC_RMID,NULL);
return 0;
}
client.c 端
#include "sys/ipc.h"
#include "sys/msg.h"
#include "sys/types.h"
#include "stdio.h"
#include "string.h"
struct mybuf
{
long mytype;
char sendbuf[128];
};
int main()
{
int key;
int msgid;
int pid;
struct mybuf wrbuf,rdbuf;
key=ftok("./a.c",'a');
if(key < 0)
{
printf("creat key failure\n");
return -1;
}
msgid=msgget(key,IPC_CREAT | 0777);
if(msgid < 0)
{
printf("creat message queue failure\n");
return -2;
}
pid=fork();
if(pid < 0)
return -3;
if(pid > 0)//write type=10
{
wrbuf.mytype=11;// voltage data
while(1)
{
memset(wrbuf.sendbuf,0,128);
printf("please input data:\n");
fgets(wrbuf.sendbuf,128,stdin);
msgsnd(msgid,(void *)&wrbuf,128,0);
}
}
//read
if(pid == 0) //read type=11
{
while(1)
{
memset(rdbuf.sendbuf,0,128);
msgrcv(msgid,(void *)&rdbuf,128,10,0);
printf("%s",rdbuf.sendbuf);
}
}
msgctl(msgid,IPC_RMID,NULL);
return 0;
}
笔记六:进程间的通信(IPC通信之消息队列)的更多相关文章
- 8.7 进程间的通讯:管道、消息队列、共享内存、信号量、信号、Socket
进程间的通讯 进程间为什么需要通讯? 共享数据.数据传输.消息通知.进程控制 进程间的通讯有哪些类型? 首先,联系前面讲过的知识,进程之间的用户地址空间是相互独立的,不能进行互相访问,但是,内核空间却 ...
- Android进阶笔记04:Android进程间通讯(IPC)之Messenger
一. Android进程间通讯之Messenger 的引入 (1)引言: 平时一说进程间通讯,大家都会想到AIDL,其实messenger和AIDL作用一样,都可以进行进程间通讯.它是基于消 ...
- 进程间的通讯(IPC)方式
内存映射 为什么要进行进程间的通讯(IPC (Inter-process communication)) 数据传输:一个进程需要将它的数据发送给另一个进程,发送的数据量在一个字节到几M字节之间共享数据 ...
- Linux学习笔记(六) 进程管理
1.进程基础 当输入一个命令时,shell 会同时启动一个进程,这种任务与进程分离的方式是 Linux 系统上重要的概念 每个执行的任务都称为进程,在每个进程启动时,系统都会给它指定一个唯一的 ID, ...
- Windows进程间通讯(IPC)----消息队列
消息队列 windows系统是通过消息驱动的,每移动一下鼠标,点击一下屏幕都会产生一个消息.这些消息会先被放在windows的一个系统消息队列(先进先出)中,windows系统会为每一个GUI线程创建 ...
- 通信有连接有消息队列选择boost.asio
通信有连接有消息队列选择boost.asio 连接自主管理 消息队列自主管理
- linux io 学习笔记(03)---共享内存,信号灯,消息队列
system V IPC 1)消息队列 2)共享内存 3)信号灯(信号量集) 1.消息队列. ipcs -q 查看系统中使用消息队列的情况 ipcrm -q +msqid 删除消息队列 消息队列工作原 ...
- Android(java)学习笔记232:Android进程间通讯(IPC)之AIDL
一.IPC inter process communication 进程间通讯 二.AIDL android interface defination language 安卓接口定义语言 满 ...
- Android(java)学习笔记175:Android进程间通讯(IPC)之AIDL
一.IPC inter process communication 进程间通讯 二.AIDL android interface defination language 安卓接口定义语言 满 ...
- Windows进程间通讯(IPC)----WM_COPYDATA
WM_COPYDATA通讯思路 通过向其他进程的窗口过程发送WM_COPYDATA消息可以实现进程间通讯. 只能通过SendMessage发送WM_COPYDATA消息,而不能通过PostMessag ...
随机推荐
- 这几年专注于流程管理与RPA落地
这些年专注于流程管理与RPA落地,乙方有些辛苦,但,也得到了客户的认可,开心. 做过多个流程管理咨询,发现信息化落地后孤岛很多,RPA 刚好可以作为各个系统的有效补充,把系统有机的结合在一起. 企业信 ...
- 过滤器函数 filtes 的使用总结
// import parseTime, formatTime and set to filter /** * Show plural label if time is plural number * ...
- Ubuntu NVIDIA显卡驱动+CUDA安装(多版本共存)
NVIDIA显卡驱动 1.禁止集成的nouveau驱动 solution 1 (recommand) # 直接移除这个驱动(备份出来) mv /lib/modules/3.0.0-12-generic ...
- Python内置函数:index
index用于返回指定值在序列的第一个位置 序列.index(值,开始索引,结束索引) 开始索引默认为0,可不传 结束索引默认为序列长度,可不传 >>> str = 'abc' &g ...
- gorm操作sqlite3,高并发读写如何避免锁库?
1. 场景 这两天一直被这个sqlit3困扰,起因是项目中需要有这样一个中间,中间件承担着API角色和流量转发的角色,需要接收来自至少300个agent的请求数据,和健康检测的请求. 所以当即想到用g ...
- 3、IOC创建对象的方法
目录 3.IOC创建对象的方法 4.Spring配置 4.1.别名 4.2.Bean的配置 4.3.import 5.依赖注入 5.1.构造器注入 5.2.Set方式注入[重点] 5.3.扩展方式注入 ...
- vsftpd配置FTP服务器(Centos7.x安装)
安装配置 1. 安装vsftpd 检查是否安装了vsftpd # rpm -qa | grep vsftpdvsftpd-2.2.2-24.el6.x86_64 如果有展示则已经安装,不需要重新安装 ...
- Hugging Face 每周速递: Chatbot Hackathon;FLAN-T5 XL 微调;构建更安全的 LLM
每一周,我们的同事都会向社区的成员们发布一些关于 Hugging Face 相关的更新,包括我们的产品和平台更新.社区活动.学习资源和内容更新.开源库和模型更新等,我们将其称之为「Hugging Ne ...
- 从零开始学习Java系列之Java运行机制与跨平台特性
全文大约[4000]字,不说废话,只讲可以让你学到技术.明白原理的纯干货!并带有丰富的案例及配图,让你更好地理解和运用文中的技术概念,给你带来具有足够启迪的思考-- 在上一篇文章中,壹哥给大家介绍了J ...
- Sentinel 流量控制
一.Sentinel 介绍 Sentinel 是阿里巴巴出品的面向分布式服务架构的轻量级流量控制组件,主要以流量为切入点,从限流,流量整形.熔断降级.系统负载保护等多个维度来保障微服务的稳定性.主页地 ...