Linux IPC实践(5) --System V消息队列(2)
消息发送/接收API
msgsnd函数
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
参数
msgid: 由msgget函数返回的消息队列标识码, 也可以是通过ipcs命令查询出来的一个已经存在的消息队列的ID号
msgp:是一个指针,指针指向准备发送的消息,
msgsz:是msgp指向的消息长度, 注意:这个长度不含保存消息类型的那个long int长整型
msgflg:控制着当前消息队列满或到达系统上限时将要发生的事情,如果msgflg = IPC_NOWAIT表示队列满不等待,返回EAGAIN错误。
消息结构在两方面受到制约: (1)它必须小于系统规定的上限值(MSGMAX); (2)它必须以一个long int长整数开始,接收者函数将利用这个长整数确定消息的类型;
//消息结构参考形式如下:
struct msgbuf
{
long mtype; /* message type, must be > 0 */
char mtext[1]; /* message data, 可以设定为更多的字节数 */
};
/**示例1:
测试1: 发送消息的最大长度为8192字节, 一旦超过这个值, 则msgsnd出错, 提示 Invalid argument错误;
测试2: 消息队列所能够接收的最大字节数16384字节, 一旦超过这个长度, 如果msgflg为0(阻塞模式), 则进程会一直阻塞下去, 直到有进程来将消息取走; 而如果msgflg为IPC_NOWAIT模式, 则一个字节也不会写入消息队列, 直接出错返回;
**/
int main(int argc, char *argv[])
{
if (argc != 3)
err_quit("Usage: ./main <type> <length>");
int type = atoi(argv[1]);
int len = atoi(argv[2]);
int msgid = msgget(0x255, 0666|IPC_CREAT);
if (msgid == -1)
err_exit("msgget error");
struct msgbuf *buf;
buf = (struct msgbuf *)malloc(len + sizeof(msgbuf::mtype));
buf->mtype = type;
if (msgsnd(msgid, buf, len, IPC_NOWAIT) == -1)
err_exit("msgsnd error");
}
msgrcv函数
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);
参数
msgid: 由msgget函数返回的消息队列标识码
msgp:是一个指针,指针指向准备接收的消息;
msgsz:是msgp指向的消息长度,这个长度不含保存消息类型的那个long int长整型
msgtype:它可以实现接收优先级的简单形式(见下图)
msgflg:控制着队列中没有相应类型的消息可供接收时将要发生的事(见下图)
返回值:
成功->返回实际放到接收缓冲区里去的字节数(注意: 此处并不包含msgbuf中的mtype的长度[man-page: msgrcv() returns the number of bytes actually copied into the mtext array.]);
失败->返回-1;
|
msgtyp参数 |
|
|
msgtyp=0 |
返回队列第一条信息 |
|
msgtyp>0 |
返回队列第一条类型等于msgtype的消息 |
|
msgtyp<0 |
返回队列第一条类型小于等于(<=)msgtype绝对值的消息,并且是满足条件的消息类型最小的消息(按照类型进行排序的顺序进行接收消息) |
|
msgflg参数 |
|
|
msgflg=IPC_NOWAIT |
队列没有可读消息不等待,返回ENOMSG错误。 |
|
msgflg=MSG_NOERROR |
消息大小超过msgsz(msgrcv 函数的第三个参数)时被截断, 并且不会报错 |
|
msgtyp>0且msgflg=MSG_EXCEPT |
接收类型不等于msgtype的第一条消息 |
/** 示例2: 消息接收(配合示例1中程序使用)
说明: -t [number], 指定接收消息的类型, 类型为number的值
-n ,指定以IPC_NOWAIT模式接收消息
**/
int main(int argc, char *argv[])
{
/** 解析参数 **/
int type = 0;
int flag = 0;
int opt;
while ((opt = getopt(argc, argv, "nt:")) != -1)
{
switch (opt)
{
case 'n': // 指定IPC_NOWAIT选项
flag |= IPC_NOWAIT;
break;
case 't': // 指定接收的类型, 如果为0的话,说明是按照顺序接收
type = atoi(optarg);
break;
default:
exit(EXIT_FAILURE);
}
}
int msgid = msgget(0x255, 0);
if (msgid == -1)
err_exit("msgget error");
const int MSGMAX = 8192; //指定一条消息的最大长度
struct msgbuf *buf;
buf = (struct msgbuf *)malloc(MSGMAX + sizeof(buf->mtype));
ssize_t nrcv;
if ((nrcv = msgrcv(msgid, buf, MSGMAX, type, flag)) == -1)
err_exit("msgrcv error");
cout << "recv " << nrcv << " bytes, type = " << buf->mtype << endl;
}
/** 综合示例: msgsnd/msgrcv, 消息发送/接收实践 **/
//1. 消息发送
int main()
{
int msgid = msgget(0x1234,0666|IPC_CREAT);
if (msgid == -1)
err_exit("msgget error");
struct msgBuf myBuffer;
for (int i = 0; i < 128; ++i)
{
myBuffer.mtype = i+1;
sprintf(myBuffer.mtext,"Hello, My number is %d",i+1);
if (msgsnd(msgid,&myBuffer,strlen(myBuffer.mtext),IPC_NOWAIT) == -1)
err_exit("msgsnd error");
}
}
//2. 消息接收:从队首不断的取数据
int main(int argc, char *argv[])
{
int msgid = msgget(0x1234, 0);
if (msgid == -1)
err_exit("msgget error");
struct msgBuf buf;
ssize_t nrcv;
while ((nrcv = msgrcv(msgid, &buf, sizeof(buf.mtext), 0, IPC_NOWAIT)) > 0)
{
cout << "recv " << nrcv << " bytes, type: " << buf.mtype
<< ", message: " << buf.mtext << endl;
}
}
[附]-getopt函数的用法
#include <unistd.h>
int getopt(int argc, char * const argv[],
const char *optstring);
extern char *optarg;
extern int optind, opterr, optopt;
//示例: 解析 ./main -n -t 3 中的参数选项
int main(int argc, char *argv[])
{
while (true)
{
int opt = getopt(argc, argv, "nt:");
if (opt == '?')
exit(EXIT_FAILURE);
else if (opt == -1)
break;
switch (opt)
{
case 'n':
cout << "-n" << endl;
break;
case 't':
int n = atoi(optarg);
cout << "-t " << n << endl;
break;
}
}
}
Linux IPC实践(5) --System V消息队列(2)的更多相关文章
- Linux IPC实践(6) --System V消息队列(3)
消息队列综合案例 消息队列实现回射客户/服务器 server进程接收时, 指定msgtyp为0, 从队首不断接收消息 server进程发送时, 将mtype指定为接收到的client进程的pid ...
- Linux IPC实践(4) --System V消息队列(1)
消息队列概述 消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法(仅局限于本机); 每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值. 消息队列也有管道一样的不足: ...
- Linux进程通信之System V消息队列
System V消息队列是Open Group定义的XSI,不属于POSIX标准.System V IPC的历史相对很早,在上个世70年代后期有贝尔实验室的分支机构开发,80年代加入System V的 ...
- linux c编程:System V消息队列一
消息队列可以认为是一个消息链表,System V 消息队列使用消息队列标识符标识.具有足 够特权的任何进程都可以往一个队列放置一个消息,具有足够特权的任何进程都可以从一个给定队列读出一个消息.在某个进 ...
- linux网络编程之system v消息队列(二)
今天继续学习system v消息队列,主要是学习两个函数的使用,开始进入正题: 下面则开始用代码来使用一下该发送函数: 在运行之前,先查看一下1234消息队列是否已经创建: 用上次编写的查看消息队列状 ...
- Linux IPC实践(13) --System V IPC综合实践
实践:实现一个先进先出的共享内存shmfifo 使用消息队列即可实现消息的先进先出(FIFO), 但是使用共享内存实现消息的先进先出则更加快速; 我们首先完成C语言版本的shmfifo(基于过程调用) ...
- Linux IPC实践(11) --System V信号量(1)
信号量API #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> int semget ...
- Linux IPC实践(9) --System V共享内存
共享内存API #include <sys/ipc.h> #include <sys/shm.h> int shmget(key_t key, size_t size, int ...
- linux网络编程之system v消息队列(一)
经过上次对于进程通讯的一些理论的认识之后,接下来会通过实验来进一步加深对进程通讯的认识,话不多说,进入正题: 其实还可以通过管道,但是,管道是基于字节流的,所以通常会将它称为流管道,数据与数据之间是没 ...
随机推荐
- VueJs(3)---V-指令
VueJs(3)---V-指令(1) 一.语法 v- 指令是带有v-的特殊属性 v-if 条件渲染 v-show v-else (必须在v-if/v-else-if/v-show指令后) v-else ...
- js密码64加密
可以在客户端对密码进行简单的64位加密,服务端对应使用64位解密即可. /** * * Base64 encode / decode * * @author * @date * @email */ f ...
- 【python标准库模块四】Json模块和Pickle模块学习
Json模块 原来有个eval函数能能够从字符串中提取出对应的数据类型,比如"{"name":"zhangsan"}",可以提取出一个字典. ...
- ACM Robot Motion
机器人已被编程为按照其指令中的路径进行操作.机器人要移动的下一个方向的指令放在网格中.可能的指令是 N north (up the page) S south (down the page) E ...
- Bootstrap3 排版-列表
无序列表 排列顺序无关紧要的一列元素. <ul> <li>...</li> </ul> 有序列表 顺序至关重要的一组元素. <ol> < ...
- 《An Industrial-Strength Audio Search Algorithm》译文
随着微信摇一摇逐渐被大众所广泛使用,听歌识曲功能也开始被关注.目前来看,像音乐雷达和微信摇一摇都采用了经典的shazam算法,为了使大家对shazam算法更加了解,我将其经典论文进行了翻译,希望对大家 ...
- The Zen Programmer
专注 何为专注 关于 休息 怎么睡觉 心无杂念 我的体会 自我分析 初学者心态 无我 不要设置职业目标 敏事慎言 正念 做自己的老板 玩物养志 结语 最近在研读Christian Grobmeier ...
- Android简易实战教程--第二十八话《加载大图片》
Android系统以ARGB表示每个像素,所以每个像素占用4个字节,很容易内存溢出.假设手机内存比较小,而要去加载一张像素很高的图片的时候,就会因为内存不足导致崩溃.这种异常是无法捕获的 内存不足并不 ...
- Android简易实战教程--第二十五话《网络图片查看器》
访问网络已经有了很成熟的框架.这一篇只是介绍一下HttpURLConnection的简单用法,以及里面的"注意点".这一篇可以复习或者学习HttpURLConnection.han ...
- Jstorm与RocketMQ整合
如果是经常关注阿里巴巴的朋友们,看到我这篇博客的题目,就知道我在参加今年的中间件比赛. 好了,废话不说,开始了. 首先我们知道,rocketmq的consumer有两种,一种是DefaultMQPus ...