精通Unix下C语言编程与项目实践》之八 消息队列发送模型

简介:



《精通Unix下C语言编程与项目实践》之八

消息队列发送模型 
作者:朱云翔,胡平

12.4.2 消息队列发送模型

本处设计一个小例子描述发送消息的实际步骤,本例要求以阻塞方式向消息队列(关键字为KEY)中写入字符串“Hello Unix!”,消息类型为TYPE
全部过程共分为5个步骤:

1. 定义消息结构

参照代码12-4,定义以下消息结构:
struct msgbuf
{
    long    mtype;      /* 消息类型 */
    char    ctext[100]; /* 消息数据 */
};

2. 打开(创建)消息队列

int msgid;
msgid = msgget(KEY, 0666|IPC_CREAT);
if (msgid < 0) 打开(创建消息失败)
如果msgid值非负,表示打开(或创建)消息队列成功。

3. 组装消息

设置消息类型和拷贝消息数据:
struct msgbuf buf;                  /* 申请消息缓冲 */
buf.mtype = 100;                    /* 设置消息类型 */
strcpy(buf.ctext, "HELLO Unix!");   /* 拷贝消息数据 */

4. 发送消息

万事俱备,只欠东风,调用函数msgsnd完成消息发送:
int ret;
ret = msgsnd(msgid, &msg, strlen(msg.ctext), 0);
注意,参数msgsz只是消息数据的长度,不包含消息类型长度。

5. 发送判断

在计算机程序设计中,售后处理非常重要,如下所示:
if (ret == -1)
{
    if (errno == EINTR) 信号中断,重新发送;
    else 系统错误
}
进程在发送消息过程中如果接收到信号,将中止消息发送并返回EINTR错误,此时重新发送即可。

实例

本处设计一个消息发送的例子,它循环读取键盘输入,并将输入的字符串信息写入到消息队列(关键字为0x1234)中,如代码12-6所示:
代码12-6 消息发送程序(节自/code/chapter12/msg1.c
#include <sys/msg.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <stdio.h>
#include <sys/errno.h>
extern int errno;
struct mymsgbuf             /* 定义消息结构 */
{
    long    mtype;          /* 消息类型 */
    char    ctext[100];     /* 消息数据 */
};
void main()
{
    struct mymsgbuf buf;    /* 申请消息缓冲 */
    int msgid;
    /* 打开(或创建)消息队列 */
    if ((msgid = msgget(0x1234, 0666|IPC_CREAT)) < 0)
    {
        fprintf(stderr, "open msg %X failed.\n", 0x1234);
        return ;
    }
    while (strncmp(buf.ctext, "exit", 4))
    {
        memset(&buf, 0, sizeof(buf));
        /* 从键盘输入消息数据内容 */
        fgets(buf.ctext, sizeof(buf.ctext), stdin);
        /* 设置消息类型为进程ID */
        buf.mtype = getpid();      
        /* 发送消息 */
        while ((msgsnd(msgid, &buf, strlen(buf.ctext), 0)) < 0)
        {
            if (errno == EINTR) continue;   /* 信号中断,重新发送 */
            return;
        }
    }
}
编译和运行代码12-6    
# make msg1
        cc -O -o msg1 msg1.c
# ./msg1
Please input:Hello World!
Please input:Hello Every Body!
Please input:读者朋友们,你们好!
Please input:exit
函数strncmp对字符串大小进行判断,当两个字符串参数完全相等时返回0,否则返回其它值。
程序调用fgets直接使用消息数据缓冲区存储输入,请确保输入字符串不要超过最大缓冲区限制,即本例中定义的99个字符(最后一个存储ASCII码的“0”)当用户输入exit时程序结束。上例中共发送了4条消息。
执行ipcs命令查询消息发送信息:
# ipcs -a -q
T     ID     KEY        MODE       OWNER    GROUP  CREATOR   CGROUP CBYTES  QNUM QBYTES LSPID LRPID  STIME    RTIME    CTIME
Message Queues:
q    162 0x00001234 --rw-rw-rw-     root      sys     root      sys     57     4  65532  1285     0 12:08:45 no-entry 12:08:28
如上所示,CBYTES值为57QNUM值为4,代表我们成功的写入了4条消息共57个字符信息。请确保执行程序msg1前消息队列0x1234不存在,或队列为空,否则显示的结果将不一致。



 本文转自 zhuyunxiang 51CTO博客,原文链接:http://blog.51cto.com/zhuyunxiang/135721,如需转载请自行联系原作者


相关文章
|
Unix Shell API
组合思维:Unix 哲学到底给现代编程带来哪些重要启示?
Unix哲学提供了一套简洁而强大的设计理念,这些理念在现代编程中依然具有重要的指导意义。通过模块化设计、组合工具、避免过早优化以及注重可复用性和可扩展性,开发者可以构建出更高效、更健壮的软件系统。希望本文能够帮助读者深入理解Unix哲学,并在实际开发中应用这些宝贵的设计原则。
390 25
|
消息中间件 Linux
Linux:进程间通信(共享内存详细讲解以及小项目使用和相关指令、消息队列、信号量)
通过上述讲解和代码示例,您可以理解和实现Linux系统中的进程间通信机制,包括共享内存、消息队列和信号量。这些机制在实际开发中非常重要,能够提高系统的并发处理能力和数据通信效率。希望本文能为您的学习和开发提供实用的指导和帮助。
1002 20
|
消息中间件 存储 NoSQL
剖析 Redis List 消息队列的三种消费线程模型
Redis 列表(List)是一种简单的字符串列表,它的底层实现是一个双向链表。 生产环境,很多公司都将 Redis 列表应用于轻量级消息队列 。这篇文章,我们聊聊如何使用 List 命令实现消息队列的功能以及剖析消费者线程模型 。
剖析 Redis List 消息队列的三种消费线程模型
|
存储 编译器 C语言
【C语言】数据类型全解析:编程效率提升的秘诀
在C语言中,合理选择和使用数据类型是编程的关键。通过深入理解基本数据类型和派生数据类型,掌握类型限定符和扩展技巧,可以编写出高效、稳定、可维护的代码。无论是在普通应用还是嵌入式系统中,数据类型的合理使用都能显著提升程序的性能和可靠性。
704 8
|
C语言 开发者
C语言中的模块化编程思想,介绍了模块化编程的概念、实现方式及其优势,强调了合理划分模块、明确接口、保持独立性和内聚性的实践技巧
本文深入探讨了C语言中的模块化编程思想,介绍了模块化编程的概念、实现方式及其优势,强调了合理划分模块、明确接口、保持独立性和内聚性的实践技巧,并通过案例分析展示了其应用,展望了未来的发展趋势,旨在帮助读者提升程序质量和开发效率。
859 5
|
C语言
C语言编程中,错误处理至关重要,能提升程序的健壮性和可靠性
C语言编程中,错误处理至关重要,能提升程序的健壮性和可靠性。本文探讨了C语言中的错误类型(如语法错误、运行时错误)、基本处理方法(如返回值、全局变量、自定义异常处理)、常见策略(如检查返回值、设置标志位、记录错误信息)及错误处理函数(如perror、strerror)。强调了不忽略错误、保持处理一致性及避免过度处理的重要性,并通过文件操作和网络编程实例展示了错误处理的应用。
502 4
|
存储 算法 Linux
C语言 多进程编程(一)进程创建
本文详细介绍了Linux系统中的进程管理。首先,文章解释了进程的概念及其特点,强调了进程作为操作系统中独立可调度实体的重要性。文章还深入讲解了Linux下的进程管理,包括如何获取进程ID、进程地址空间、虚拟地址与物理地址的区别,以及进程状态管理和优先级设置等内容。此外,还介绍了常用进程管理命令如`ps`、`top`、`pstree`和`kill`的使用方法。最后,文章讨论了进程的创建、退出和等待机制,并展示了如何通过`fork()`、`exec`家族函数以及`wait()`和`waitpid()`函数来管理和控制进程。此外,还介绍了守护进程的创建方法。
C语言 多进程编程(一)进程创建
|
NoSQL C语言 索引
十二个C语言新手编程时常犯的错误及解决方式
C语言初学者常遇错误包括语法错误、未初始化变量、数组越界、指针错误、函数声明与定义不匹配、忘记包含头文件、格式化字符串错误、忘记返回值、内存泄漏、逻辑错误、字符串未正确终止及递归无退出条件。解决方法涉及仔细检查代码、初始化变量、确保索引有效、正确使用指针与格式化字符串、包含必要头文件、使用调试工具跟踪逻辑、避免内存泄漏及确保递归有基准情况。利用调试器、编写注释及查阅资料也有助于提高编程效率。避免这些错误可使代码更稳定、高效。
2338 12
|
算法 Unix 数据安全/隐私保护
Python编程--UNIX口令破解机
Python编程--UNIX口令破解机
240 1
|
Linux C语言
C语言 多进程编程(三)信号处理方式和自定义处理函数
本文详细介绍了Linux系统中进程间通信的关键机制——信号。首先解释了信号作为一种异步通知机制的特点及其主要来源,接着列举了常见的信号类型及其定义。文章进一步探讨了信号的处理流程和Linux中处理信号的方式,包括忽略信号、捕捉信号以及执行默认操作。此外,通过具体示例演示了如何创建子进程并通过信号进行控制。最后,讲解了如何通过`signal`函数自定义信号处理函数,并提供了完整的示例代码,展示了父子进程之间通过信号进行通信的过程。