嵌入式学习第二十四天!(进程间通信:消息队列、共享内存、信号灯)

本文主要是介绍嵌入式学习第二十四天!(进程间通信:消息队列、共享内存、信号灯),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

进程间的通信:

  消息队列共享内存信号灯

     1. IPC对象:内存文件

        1. ipcs:

            查看系统中的消息队列,共享内存、信号灯的信息

        2. ipcrm:

            删除消息队列、共享内存、信号灯

ipcrm -Q/-M/-S key
ipcrm -q/-m/-s 消息对立ID/共享内存ID/信号灯ID

1. 消息队列:

    1. 操作流程:

        创建消息队列 -> 发送消息 -> 接收消息

    2. 函数接口:

        1. ftok:
key_t ftok(const char *pathname, int proj_id);

          功能:根据pathname和proj_id生成一个key_t类型的key值,将来可以创建消息队列、共享内存、信号灯

          参数:

              pathename:文件路径

              proj_id:8位非0值

          返回值:

              成功返回key_t类型的IPC对象的key值
              失败返回-1 

        2. msgsnd:
int msgget(key_t key, int msgflg);

          功能:根据key值对象的IPC对象创建一个消息队列

          参数:

              key:IPC对象名字

              msgflg:IPC_CREAT    对象不存在就创建

                            IPC_EXCL       对象存在报错

                            IPC_CREAT | 0664

          返回值:

              成功返回消息队列ID
              失败返回-1

        3.  msgsnd:
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);

          功能:向消息队列中发送消息

          参数:

              msqid:消息队列的ID号

              msgp:发送消息空间的首地址

struct msgbuf
{int mtypes;    /* message type, must be > 0 */char mtext[1]  /* message data */
};

              msgz:发送消息内容的大小(不包含发送消息类型)

              msgflg:属性,默认位0

          返回值:

              成功返回0 
              失败返回-1 

        4. msgrcv:
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);

          功能:从消息队列中接收消息

          参数:

              msqid:消息队列的ID号

              msgp:存放接收到消息空间的首地址

              msgsz:最多接收消息的空间大小

              msgtype:想要接收消息的类型

              msgflg:属性,默认为0

          返回值:

              成功返回实际接收的字节数
              失败返回-1 

        5. msgctl:
int msgctl(int msqid, int cmd, struct msqid_ds *buf);

          功能:向消息队列发送一条cmd命令

          参数:

              msqid:消息队列的ID号

              cmd:IPC_RMID  删除消息队列

              buf:默认传NULL

        练习:

          利用消息队列实现clientA和clientB两个进程任务的全双工聊天功能

clientA.c

 #include "head.h"key_t key;
int msgid = 0;
pthread_t atob;
pthread_t btoa;struct msgbuf
{long mtype;char mtext[256];
};void *thread_atob(void *arg)
{struct msgbuf sndmsg;while(1){memset(&sndmsg, 0, sizeof(sndmsg));sndmsg.mtype = 100;gets(sndmsg.mtext);msgsnd(msgid, &sndmsg, sizeof(struct msgbuf)-sizeof(long), 0);if(strcmp(sndmsg.mtext, ".quit") == 0){break;}}pthread_cancel(atob);return NULL;}void *thread_btoa(void *arg)
{struct msgbuf recmsg;while(1){memset(&recmsg, 0, sizeof(recmsg));msgrcv(msgid, &recmsg, sizeof(recmsg)-sizeof(long), 200, 0);printf("RECV:%s\n", recmsg.mtext);if(strcmp(recmsg.mtext, ".quit") == 0){break;}}pthread_cancel(btoa);return NULL;
}int main(void)
{pthread_create(&atob, NULL, thread_atob, NULL);pthread_create(&btoa, NULL, thread_btoa, NULL);key = ftok(".", 'a');if(key == -1){perror("fail to ftok");return -1;}msgid = msgget(key, IPC_CREAT | 0664);if(msgid == -1){perror("fail to msgget");return -1;}pthread_join(atob, NULL);pthread_join(btoa, NULL);msgctl(msgid, IPC_RMID, NULL);return 0;
}

clientB.c

#include "head.h"key_t key;
int msgid = 0;
pthread_t atob;
pthread_t btoa;struct msgbuf
{long mtype;char mtext[256];
};void *thread_btoa(void *arg)
{struct msgbuf sndmsg;while(1){memset(&sndmsg, 0, sizeof(sndmsg));sndmsg.mtype = 200;gets(sndmsg.mtext);msgsnd(msgid, &sndmsg, sizeof(struct msgbuf)-sizeof(long), 0);if(strcmp(sndmsg.mtext, ".quit") == 0){break;}}pthread_cancel(btoa);return NULL;
}void *thread_atob(void *arg)
{struct msgbuf recmsg;while(1){memset(&recmsg, 0, sizeof(recmsg));msgrcv(msgid, &recmsg, sizeof(recmsg)-sizeof(long), 100, 0);printf("RECV:%s\n", recmsg.mtext);if(strcmp(recmsg.mtext, ".quit") == 0){break;}}pthread_cancel(atob);return NULL;
}int main(void)
{pthread_create(&atob, NULL, thread_atob, NULL);pthread_create(&btoa, NULL, thread_btoa, NULL);key = ftok(".", 'a');if(key == -1){perror("fail to ftok");return -1;}msgid = msgget(key, IPC_CREAT | 0664);if(msgid == -1){perror("fail to msgget");return -1;}pthread_join(atob, NULL);pthread_join(btoa, NULL);msgctl(msgid, IPC_RMID, NULL);return 0;
}

2. 共享内存:

  进程间通信最高效的形式

  1. 操作方式:

        创建共享内存  ->  映射到共享内存中  ->  共享内存操作  ->  解除映射  ->  删除共享内存

  2. 函数接口:

        1. ftok:
key_t ftok(const char *pathname, int proj_id);

          功能:根据pathname和proj_id生成一个key_t类型的key值,将来可以创建消息队列、共享内存、信号灯

          参数:

              pathename:文件路径

              proj_id:8位非0值

          返回值:

              成功返回key_t类型的IPC对象的key值
              失败返回-1 

        2. shmget:
int shmget(key_t key, size_t size, int shmflg);

          功能:创建一个共享内存

          参数:

              key:IPC对象名称

              size:共享内存的大小

              shmflg:

                    IPC_CREAT

                    IPC_EXCL

          返回值:

              成功返回共享内存ID
              失败返回-1 

        3. shmat
void *shmat(int shmid, const void *shmaddr, int shmflg);

          功能:将一个地址映射到共享内存中

          参数:

              shmid:共享内存ID号

              shmaddr:

                    NULL:让系统选择一个合适的地址映射

                    不为NULL:shmflg设定SHM_RND选择离给定地址最近的能够映射的地址进行映射,否则传递地址为4K的整数倍

        4. shmdt:
int shmdt(const void *shmaddr);

          功能:解除映射

          参数:

              shmaddr:映射的地址

          返回值:

              成功返回0 
              失败返回-1 

        5. shmctl:
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

          功能:向共享内存发送命令

          参数:

              shmid:共享内存ID号

              cmd:IPC_RMID 删除共享内存

              buf:NULL

          返回值:

              成功返回0 
              失败返回-1

        练习:

              编写2个进程任务,write.c负责从终端接收字符串写入共享内存中,read.c负责将共享内存中的数据打印在终端

write.c

#include "head.h"int main(void)
{key_t key;int shmid;char *pshm = NULL;key = ftok(".", 'a');if(key == -1){perror("fail to ftok");return -1;}shmid = shmget(key, 4096, IPC_CREAT | 0664);if(shmid == -1){perror("fail to shmget");return -1;}pshm = (char *)shmat(shmid, NULL, 0);if(pshm == NULL){perror("fail to shmat");return -1;}while(1){gets(pshm);if(!strcmp(pshm, ".quit")){shmdt(pshm);break;}}shmctl(shmid, IPC_RMID, NULL);return 0;
}
#include "head.h"int main(void)
{key_t key;int shmid;char *pshm = NULL;char *ptmp = NULL;key = ftok(".", 'a');if(key == -1){perror("fail to ftok");return -1;}shmid = shmget(key, 4096, IPC_CREAT | 0664);if(shmid == -1){perror("fail to shmget");return -1;}pshm = shmat(shmid, NULL, 0);ptmp = malloc(4096);	strcpy(ptmp, pshm);while(1){if (strcmp(pshm, ptmp) == 0){continue;}else{printf("%s\n", pshm);strcpy(ptmp, pshm);}if(!strcmp(pshm, ".quit")){free(ptmp);shmdt(pshm);break;}}shmctl(shmid, IPC_RMID, NULL);return 0;
}

        在这里使用了一个ptmp和原来的pshm比较,从而实现了一个从终端写一个,接收一个,其实应该使用信号灯实现,但是在这里还没有学,所以先暂时使用这个来实现。如果不加判断的话,终端输出的结果如下所示:

        终端会一直持续打印共享内存中的数据,继而需要用到信号灯来实现,写一条,只接收一条

这篇关于嵌入式学习第二十四天!(进程间通信:消息队列、共享内存、信号灯)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/775226

相关文章

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Redis延迟队列的实现示例

《Redis延迟队列的实现示例》Redis延迟队列是一种使用Redis实现的消息队列,本文主要介绍了Redis延迟队列的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录一、什么是 Redis 延迟队列二、实现原理三、Java 代码示例四、注意事项五、使用 Redi

python多进程实现数据共享的示例代码

《python多进程实现数据共享的示例代码》本文介绍了Python中多进程实现数据共享的方法,包括使用multiprocessing模块和manager模块这两种方法,具有一定的参考价值,感兴趣的可以... 目录背景进程、进程创建进程间通信 进程间共享数据共享list实践背景 安卓ui自动化框架,使用的是

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

C#如何优雅地取消进程的执行之Cancellation详解

《C#如何优雅地取消进程的执行之Cancellation详解》本文介绍了.NET框架中的取消协作模型,包括CancellationToken的使用、取消请求的发送和接收、以及如何处理取消事件... 目录概述与取消线程相关的类型代码举例操作取消vs对象取消监听并响应取消请求轮询监听通过回调注册进行监听使用Wa

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

hdu1180(广搜+优先队列)

此题要求最少到达目标点T的最短时间,所以我选择了广度优先搜索,并且要用到优先队列。 另外此题注意点较多,比如说可以在某个点停留,我wa了好多两次,就是因为忽略了这一点,然后参考了大神的思想,然后经过反复修改才AC的 这是我的代码 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<