/*send.c*/

  1. /*send.c*/
  2. #include <stdio.h>
  3. #include <sys/types.h>
  4. #include <sys/ipc.h>
  5. #include <sys/msg.h>
  6. #include <errno.h>
  7.  
  8. #define MSGKEY 1024
  9.  
  10. struct msgstru
  11. {
  12. long msgtype;
  13. char msgtext[2048];
  14. };
  15.  
  16. main()
  17. {
  18. struct msgstru msgs;
  19. int msg_type;
  20. char str[256];
  21. int ret_value;
  22. int msqid;
  23.  
  24. msqid=msgget(MSGKEY,IPC_EXCL); /*检查消息队列是否存在*/
  25. printf("msqid:%d \n",msqid);
  26. if(msqid < 0){
  27. msqid = msgget(MSGKEY,IPC_CREAT|0666);/*创建消息队列*/
  28. printf("msqid:%d \n",msqid);
  29. if(msqid <0){
  30. printf("failed to create msq | errno=%d [%s]\n",errno,strerror(errno));
  31. exit(-1);
  32. }
  33. }
  34.  
  35. while (1){
  36. printf("input message type(end:0):");
  37. scanf("%d",&msg_type);
  38. if (msg_type == 0)
  39. break;
  40. printf("input message to be sent:");
  41. scanf ("%s",str);
  42. msgs.msgtype = msg_type;
  43. strcpy(msgs.msgtext, str);
  44. /* 发送消息队列 */
  45. printf("msqid:%d,msgs.msgtext: %s,size:%d\n",msqid,msgs.msgtext,sizeof(msgs.msgtext));
  46. ret_value = msgsnd(msqid,&msgs,sizeof(struct msgstru),IPC_NOWAIT);
  47. printf("sendXxdl(),ret_value:%d,sizeof(struct msgstru):%d \n",ret_value,sizeof(struct msgstru));
  48. if ( ret_value < 0 ) {
  49. printf("msgsnd() write msg failed,errno=%d[%s]\n",errno,strerror(errno));
  50. exit(-1);
  51. }
  52. }
  53. msgctl(msqid,IPC_RMID,0); //删除消息队列
  54. }

/*receive.c */

  1. /*receive.c */
  2. #include <stdio.h>
  3. #include <sys/types.h>
  4. #include <sys/ipc.h>
  5. #include <sys/msg.h>
  6. #include <errno.h>
  7.  
  8. #define MSGKEY 1024
  9.  
  10. struct msgstru
  11. {
  12. long msgtype;
  13. char msgtext[2048];
  14. };
  15.  
  16. /*子进程,监听消息队列*/
  17. void childproc(){
  18. struct msgstru msgs;
  19. int msgid,ret_value;
  20. char str[512];
  21.  
  22. while(1){
  23. msgid = msgget(MSGKEY,IPC_EXCL );/*检查消息队列是否存在 */
  24. if(msgid < 0){
  25. printf("msq not existed! errno=%d [%s]\n",errno,strerror(errno));
  26. sleep(2);
  27. continue;
  28. }
  29. /*接收消息队列*/
  30. ret_value = msgrcv(msgid,&msgs,sizeof(struct msgstru),0,0);
  31. printf("text=[%s] pid=[%d]\n",msgs.msgtext,getpid());
  32. }
  33. return;
  34. }
  35.  
  36. void main()
  37. {
  38. int i,cpid;
  39. printf("open receive\n");
  40. /* create 5 child process */
  41. for (i=0;i<1;i++){
  42. cpid = fork();
  43. if (cpid < 0)
  44. printf("fork failed\n");
  45. else if (cpid ==0) /*child process*/
  46. childproc();
  47. }
  48. }

实现linux进程通信的方式有5种:
--信号(Singal)
--管道(Pipe)
--消息队列(Message)
--信号量(Semaphore)

消息队列
一、概念
消息队列就是一个消息的链表。可以把消息看作一个记录,具有特定的格式以及特定的优先级。对消息队列有写权限的进程可以向中按照一定的规则添加新消息;有读权限的进程则可以读走消息。读走就没有了。消息队列是随内核持续的。 只有内核重启或人工删除时,该消息才会被删除。在系统范围内,消息队列与键值唯一对应。

消息队列的数据结构
1.struct msqid_ds *msgque[MSGMNI]向量:
msgque[MSGMNI]是一个msqid_ds结构的指针数组,每个msqid_ds结构指针代表一个系统消息队列,msgque[MSGMNI]的大小为MSGMNI=128,也就是说系统最多有MSGMNI=128个消息队列
2.struct msqid_ds
一个消息队列结构
struct msqid_ds 中主要数据成员介绍如下:
struct msqid_ds
{
struct ipc_perm msg_perm;
        struct msg *msg_first;                                     /*消息队列头指针*/
        struct msg *msg_last;                                     /*消息队列尾指针*/
        __kernel_time_t msg_stime;                          /*最后一次插入消息队列消息的时间*/
        __kernel_time_t msg_rtime;                          /*最后一次接收消息即删除队列中一个消息的时间*/
        __kernel_time_t msg_ctime; 
        struct wait_queue *wwait;                                /*发送消息等待进程队列*/
        struct wait_queue *rwait;  
        unsigned short msg_cbytes;
        unsigned short msg_qnum;                             /*消息队列中的消息个数*/
        unsigned short msg_qbytes;
        __kernel_ipc_pid_t msg_lspid;                        /*最后一次消息发送进程的pid*/
        __kernel_ipc_pid_t msg_lrpid;                         /*最后一次消息发送进程的pid*/
};
3.struct msg 消息节点结构:
msqid_ds.msg_first,msg_last维护的链表队列中的一个链表节点
struct msg
{
       msg *msg_next;              /*下一个msg*/
       long msg_type;                /*消息类型*/
       *msg_spot;                       /*消息体开始位置指针*/
        msg_ts;                           /*消息体长度*/
        message;                     /*消息体*/
}
4.msgbuf消息内容结构:
msg 消息节点中的消息体,也是消息队列使用进程(消息队列发送接收进程)发送或者接收的消息
struct msgbuf
{
        long mtype;        --消息类型
        char mtext[n];     --消息内容
}

二、步骤及思路
1、取得键值
2、打开、创建消息队列
3、发送消息
4、接收消息
下面具体看看:
1、取得键值
      key_t ftok(char *pathname, char proj)
头文件为<sys/ipc.h>。返回文件名对应的键值,失败返回 -1。proj是项目名,随便写,不为0就行。
fname就是你指定的文件名(已经存在的文件名)。需要有-t 的权限,或用root权限执行,通常设为/tmp或设为" . "。这里我感觉不用这个函数也行,因为key值可以自己指定,例如: #define KEY_MSG 0x101
2、打开、创建消息队列
     int msgget(key_t key, int msgflg)
头文件为<sys/msg.h>。key由ftok获得。
msgflg有:
IPC_CREAT 创建新的消息队列,应配有文件权限0666。
IPC_EXCL  与IPC_CREAT一同使用,表示如果要创建的消息队列已经存在,则返回错误。
IPC_NOWAIT 读写消息不阻塞。
当没有与key相对应的消息队列并且msgflg中包含了IPC_CREAT标志 或 key的参数为IPC_PRIVATE 时,创建一个新的消息队列。
3、发送消息
         int msgsnd(int msqid, struct msgbuf *msgp, int msgsz, int msgflg)
向消息队列发送一条消息。msqid为消息队列的id,msgp为存放消息的结构体。msgsz是消息的长度,和数组的大小不一样哦。msgflg为消息标志,通常为0,也可以为IPC_NOWAIT。出错返回 -1。
消息格式
      struct msgbuf  {
                   long mtype;
                   char mtext[100];
      };
4、接收消息
int msgrcv(int msqid, struct msgbuf *msgp, int msgsz, long msgtyp, int msgflg)
从msqid代表的消息队列中读取一个msgtyp类型的消息,并把消息存储在msgp指定的msgbuf结构中。读取后队列中的消息将会删除。size为结构体中数据的大小,不要计算msgtyp。出错返回 -1。

Linux:进程通信之消息队列Message实例的更多相关文章

  1. linux进程通信:消息队列

    消息队列可以实现两个没有关系的进程之间的通信. 创建了一个消息队列后,进程可以往里面放消息,也可以取消息.因为这个消息队列是有名字的,所以就算是两个没有关系的进程,也能通信. 而且人性化的一点是,可以 ...

  2. linux 进程学习笔记-消息队列messagequeue

    可以想象,如果两个进程都可以访问同一个队列:其中一个进程(sender)向其中写入结构化数据,另外一个进程(receiver)再从其中把结构化的数据读取出来.那么这两个进程就是在利用这个队列进行通信了 ...

  3. [转]Linux进程通信之POSIX消息队列

    进程间的消息队列可以用这个实现,学习了下. http://blog.csdn.net/anonymalias/article/details/9799645?utm_source=tuicool&am ...

  4. linux消息队列编程实例

    转自:linux 消息队列实例 前言: 消息队列就是一个消息的链表.可以把消息看作一个记录,具有特定的格式以及特定的优先级.对消息队列有写权限的进程可以向其中按照一定的规则添加新消息:对消息队列有读权 ...

  5. Linux 进程间通信(posix消息队列 简单)实例

    Linux 进程间通信(posix消息队列 简单)实例 详情见: http://www.linuxidc.com/Linux/2011-10/44828.htm 编译: gcc -o consumer ...

  6. Linux进程间通信(System V) --- 消息队列

    消息队列 IPC 原理 消息队列是消息的链式队列,如下图为消息队列的模型.整个消息队列有两种类型的数据结构. 1.msqid_ds 消息队列数据结构:描述整个消息队列的属性,主要包括整个消息队列的权限 ...

  7. linux进程通信

    e14: 进程间通信(进程之间发送/接收字符串/结构体): 传统的通信方式: 管道(有名管道 fifo,无名管道 pipe) 信号 signal System V(基于IPC的对象):         ...

  8. IPC通信:Posix消息队列

    IPC通信:Posix消息队列 消息队列可以认为是一个链表.进程(线程)可以往里写消息,也可以从里面取出消息.一个进程可以往某个消息队列里写消息,然后终止,另一个进程随时可以从消息队列里取走这些消息. ...

  9. Linux进程间通信:管道,信号量,消息队列,信号,共享内存,套接字

    Linux下的进程通信手段基本上是从UNIX平台上的进程通信手段继承而来的.而对UNIX发展做出重大贡献的两大主力AT&T的贝尔实验室及BSD(加州大学伯克利分校的伯克利软件发布中心)在进程间 ...

随机推荐

  1. EXCEL技能之数据去重

    本篇不属于技术类博文,只是想找个地方记录而已,既然是我的博客嘛,那就自己想写什么就写什么了. CRM中有个EXCEL数据导入功能,几千条数据导入CRM后去重,那是死的心都有的.往回想想EXCEL是否有 ...

  2. SSH架构实现在线支付功能

    在线支付是指卖方与卖方通过因特网上的电子商务网站进行交易时,银行为其提供网上资金结算服务的一种业务,她为企业和个人提供了一个安全.快捷.方便的电子商务应用环境和网上资金结算工具,在线支付不仅帮助企业实 ...

  3. Servlet之文件上传

    上传表单中的注意事项: 表单 method 属性应该设置为 POST 方法,不能使用 GET 方法 表单 enctype 属性应该设置为multipart/form-data 下面的实例是借助于com ...

  4. 使用Swift开发一个MacOS的菜单状态栏App

    猴子原创,欢迎转载.转载请注明: 转载自Cocos2Der-CSDN,谢谢! 原文地址: http://blog.csdn.net/cocos2der/article/details/52054107 ...

  5. ubuntu中安装samba

    为了方便的和Windows之间进行交互,samba必不可少. 当然,他的安装使用也很简单: 安装: sudo apt-get install samba sudo apt-get install sm ...

  6. SpriteBuilder&Cocos2D使用CCEffect特效实现天黑天亮过度效果

    大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) 在动作或RPG类游戏中我们有时需要天黑和天亮过度的效果来完成场 ...

  7. iOS动画进阶 - 教你写 Slack 的 Loading 动画

    (转载自:http://blog.csdn.net/wang631106979/article/details/52473985) 如果移动端访问不佳,可以访问我的个人博客 前几天看了一篇关于动画的博 ...

  8. Mybatis源码之Statement处理器PreparedStatementHandler(五)

    PreparedStatementHandler就是调用PreparedStatement来执行SQL语句,这样在第一次执行sql语句时会进行预编译,在接下来执行相同的SQL语句时会提高数据库性能 源 ...

  9. J2EE学习从菜鸟变大鸟之四 JNDI(Java Naming and Directory Interface)

    掌握 J2EE 是件不是很轻松的事哈,但是很有意思,抽象抽象哈哈,因为它包含的技术和缩略语在不断地增长.Java 命名和目录接口(Java Naming and Directory Interface ...

  10. Framework启动过程浅析

    浅显的总结一下Framework启动大概过程 总体 Android底层是linux系统,因而在开机时仍然是运行天字第一号进程inti,读取init.rc来创建第一个Dalvik进程zygote,下面是 ...