System V IPC(1)-消息队列
一.概述
System V三种IPC:消息队列,信号量,共享内存。这三种IPC最先出现在AT&T System v UNIX上面,并遵循XSI标准,有时候也被称为XSI IPC。
本文先探讨消息队列:
1.消息队列允许进程以消息的形式交换数据。读写都是针对整条消息,不能读写消息的一部分,不像管道那样可以以流的形式读写任意字节。
2.消息队列除了包含数据外,还有一个整数来表示该消息的类型。读取消息的时候即可以按照先进先出方式读取,也可以按照消息类型来读取。
二.函数接口
1.创建一个消息队列
#include <sys/msg.h> int msgget(key_t key, int msgflg);
key:是一个整数,该函数会将key转换成一个IPC标识符。key有3种方法定义:1.手动随意指定一个整数。2.把IPC_PRIVATE当作key传入,系统会自动生成。3.用ftok()函数。
msgflg:指定该消息的权限,跟文件的权限控制类似。
2.发送消息
#include <sys/msg.h> int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
msqid:用msget()获取的id。
mgsp:存储消息的结构指针,下面的mtype就是自定义的消息类型,mtext是消息数据。
struct msgbuf {
long mtype; /* message type, must be > 0 */
]; /* message data */
};
msgsz:消息的大小,对应上面msgbuf里面的mtext。
msgflg:控制消息发送时异常状况,如消息队列满。
3.接收消息
#include <sys/msg.h> ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);
msqid:用msget()获取的id,或者已知的消息ID。
msgp,msgsz:同msgsnd()。
msgtyp:接收消息的类型,即msgbuf里面的mtype。但还有别的用法:
如果为0,就获取队列中第一个可用消息。
大于0,获取相同类型消息的第一个,即mtype。
小于0,获取等于或小于mtype的绝对值第一个消息。等会我们一一做实验。
msgflg:同msgsnd()。
4.消息控制
#include <sys/msg.h> int msgctl(int msqid, int cmd, struct msqid_ds *buf);
cmd:有3个选项,IPC_STAT,IPC_SET,IPC_RMID。前2个是获取和设置msgid对应的消息结构体,最后一个是删除消息队列。
三.简单的例子
1.创建消息队列
/**
* @file msg_create.c
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/msg.h>
void err_exit(const char *err_msg)
{
printf("%s error\n", err_msg);
exit();
}
int main(void)
{
| IPC_CREAT);
)
err_exit("msgget()");
printf("create msg_id:%d\n", msg_id);
;
}
2.发送消息
/**
* @file msg_send.c
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/msg.h>
#define MAX_BUFFER 1024
typedef struct
{
long msg_type;
char msg_text[MAX_BUFFER];
} msg_t;
void err_exit(const char *err_msg)
{
printf("%s error\n", err_msg);
exit();
}
int main(int argc, char *argv[])
{
)
{
printf(]);
exit();
}
]);
msg_t send_msg;
];
int text_len = strlen(text);
send_msg.msg_type = atoi(argv[]);
memcpy((void *)send_msg.msg_text, text, text_len);
) == -)
err_exit("msgsnd()");
;
}
3.接收消息
/**
* @file msg_recv.c
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/msg.h>
#define MAX_BUFFER 1024
typedef struct
{
long msg_type;
char msg_text[MAX_BUFFER];
} msg_t;
void err_exit(const char *err_msg)
{
printf("%s error\n", err_msg);
exit();
}
int main(int argc, const char *argv[])
{
)
{
printf(]);
exit();
}
]);
msg_t recv_msg;
]);
) == -)
err_exit("msgrcv()");
printf("receive:%s\n", recv_msg.msg_text);
//if (msgctl(msg_id, IPC_RMID, 0) == -1)
// err_exit("msgctl()");
;
}
四.实验
1.创建消息,编译执行msg_create.c,用ipcs -q查看消息:

可以看到:msqid就是用IPC_PRIVATE当作key传入,系统会自动生成的,msqid=262144等会接收消息要用。perms是我们代码设置的权限,此时的消息字节和消息数都是0。
2.发送消息,编译执行msg_send.c,并发消息,用ipcs -q查看消息:

上面./mes_send后面依次是:刚刚创建的消息队列id,消息类型,消息数据。
接下来,我们再继续发送1条1类型消息,2条2类型消息,2条3类型消息,等会接收消息做实验。

现在我们有6条消息了。
3.接收消息,编译msg_recv.c。我们主要来实验msgrcv()里面的msg_type参数。即该文件的第36行代码。
3.1:当msg_type等于0时,获取队列中第一个可用消息。

可以看到,1234就是我刚刚第一次发送到该队列的消息。
3.2:当msg_type大于0,获取具有相同类型的第一个消息:

上面我们获取的是3类型的消息,接收的刚好是第一次发送3号消息的haha。
3.3:当msg_type小于0,获取等于或小于msg_type绝对值的第一个消息:

上面,-3的绝对值是3,而队列中存在最先放进去的消息是1号消息22222(本来是1号的1234,刚刚我们做实验时读取走了,所以剩下它第一)。1小于3,所以1号消息被读取。
4.消息队列的删除,msg_recv.c里面第41行代码,如果放开后编译执行,收到一条消息后整个队列全部删除。
System V IPC(1)-消息队列的更多相关文章
- System V IPC 之消息队列
消息队列和共享内存.信号量一样,同属 System V IPC 通信机制.消息队列是一系列连续排列的消息,保存在内核中,通过消息队列的引用标识符来访问.使用消息队列的好处是对每个消息指定了特定消息类型 ...
- 四十九、进程间通信——System V IPC 之消息队列
49.1 System V IPC 介绍 49.1.1 System V IPC 概述 UNIX 系统存在信号.管道和命名管道等基本进程间通讯机制 System V 引入了三种高级进程间通信机制 消息 ...
- Linux 系统编程 学习:04-进程间通信2:System V IPC(1)
Linux 系统编程 学习:04-进程间通信2:System V IPC(1) 背景 上一讲 进程间通信:Unix IPC-信号中,我们介绍了Unix IPC中有关信号的概念,以及如何使用. IPC的 ...
- Linux 系统编程 学习:05-进程间通信2:System V IPC(2)
Linux 系统编程 学习:05-进程间通信2:System V IPC(2) 背景 上一讲 进程间通信:System V IPC(1)中,我们介绍了System IPC中有关消息队列.共享内存的概念 ...
- 第3章 System V IPC
3.1 概述 System V IPC 包含:System V消息队列.System V信号量.System V共享内存. 3.2 key_t 键和 ftok函数 这三种类型的System V IPC ...
- 《Unix网络编程》卷2 读书笔记 第3章- System V IPC
1. 概述 三种类型的System V IPC:System V 消息队列.System V 信号量.System V 共享内存区 System V IPC在访问它们的函数和内核为它们维护的信息上共享 ...
- 从并发处理谈PHP进程间通信(二)System V IPC
.container { margin-right: auto; margin-left: auto; padding-left: 15px; padding-right: 15px } .conta ...
- System V IPC 之共享内存
IPC 是进程间通信(Interprocess Communication)的缩写,通常指允许用户态进程执行系列操作的一组机制: 通过信号量与其他进程进行同步 向其他进程发送消息或者从其他进程接收消息 ...
- System V IPC
1.概述 System V IPC共有三种类型:System V消息队列.System V 信号量.System V 共享内存区. System V IPC操作函数如下: 2.key_t键和ftok函 ...
随机推荐
- SpringFramework的简介
一.前言 Spring提供了一种轻量级的解决方案,用于建立"快装式企业应用".在此基础上,Spring还提供了包括声明式事务管理,RMI或Web Services远程访问业务逻辑, ...
- ubuntu 14.04 解决JavaMelody 图片中文乱码
从windows系统中,copy了MSYH.TTC和MSYHBD.TTC 2个文件到 服务器的%JAVA_HOME%\jre\lib\fonts\fallback 目录中, (如果fallback目录 ...
- JavaScript学习笔记-实现枚举类型,扑克牌应用
//实现枚举类型,扑克牌应用 function creatEnum(p){ //构造函数 var Enumeration = function(){throw 'can not Ins ...
- Snort - manual 笔记(五)
1.9 Miscellaneous 1.9.1 Running Snort as a Daemon 如果你想让Snort作为守护程序运行,你可以在最后加上 -D 选项.清注意如果你想通过发送一个 SI ...
- 2013 Visual Studio Magazine读者选择奖界面框架类获奖情况
2013 Visual Studio Magazine读者选择奖已经正式揭晓了!据了解,截至今年此奖项已经评选了21次,非常值得.NET开发人员信赖和参考.此次评选共有400多个产品角逐28个分类的奖 ...
- AWS EC2 复制实例后,自定义指标无法显示数据
从一个实例创建了一个AMI,然后通过这个AMI创建新的EC2实例,结果发票自定义指标不会显示: 系统一直在邮件中提示: print() on closed filehandle MDATA at Cl ...
- Android 中的Json解析工具fastjson 、序列化、反序列化
Android中通常需要访问服务器,然而服务器返回的数据很多时候都是Json格式 1.fastjson简介 阿里巴巴FastJson是一个Json处理工具包,包括“序列化”和“反序列化”两部分,它具备 ...
- 数据持久化(一)--NSKeyedArchiver
数据持久化: 将内存中的数据按某种格式存进磁盘 数据的种类: 1,结构化的数据 2, 字节流数据 结构化的数据 字节流 内存中结构化的数据 -> 磁盘, 叫: 归档 字 ...
- Android Frameworks层介绍
Activity Manager用来管理应用程序生命周期并提供常用的导航回退功能. Window Manager提供一些我们访问手机屏幕的方法.屏幕的透明度.亮度.背景. Content Provid ...
- Xcode常用快捷键的使用
熟练使用Xcode的一些快捷方式,会大大加快项目开发的速度.