IPC对象、消息队列 、共享内存

news/2024/7/21 6:53:31 标签: linux, vim, 进程间通信, 消息队列

我要成为嵌入式高手之3月4日Linux高编第十四天!!

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

一、IPC对象

内存文件,如何查看?

1、ipcs:

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

2、ipcrm:

        功能:删除消息队列或者共享内存或者信号灯

        ipcrm  -Q/-M/-S  key(ipc对象的名字)

        ipcrm  -q/-m/-s  消息队列的id/共享内存的id/信号灯的id

二、消息队列相关的函数接口

消息队列的操作流程:

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

1、ftok

#include <sys/types.h>
#include <sys/ipc.h>

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

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

参数:

        pathname:文件路径

        proj_id:8位非0值

返回值:

        成功返回key_t类型的IPC对象的key值

        失败返回-1;

2、msgget

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

int msgget(key_t key, int msgflg);

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

参数:

        key:IPC对象的名字

        msgflg: IPC_CREAT        对象不存在创建

                       IPC_EXCL           对象存在报错

                       IPC_CREAT || 0664加权限

返回值:

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

3、msgsnd

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

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

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

参数:

        msqid:消息队列的id号

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

         struct msgbuf {
               long mtype;       /* 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:存放接收到消息空间的首地址

         struct msgbuf {
               long mtype;       /* message type, must be > 0 */(消息的类型)
               char mtext[1];    /* message data */(消息的内容)
           };

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

        msgtyp:相要接收消息的类型

        msgflg:属性,默认填0

返回值:

        成功返回实际接受的字节数,失败返回-1;

5、msgctl

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

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

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

参数:

        msgqid:消息队列的id号

        cmd:命令:IPC_RMID 删除消息队列

         buf:默认传NULL

返回值:成功返回0,失败返回-1;

练习:

        编写两个进程任务(read.c   send.c),send.c从终端接收一个字符串利用消息队列发送给recv.c,recv.c从消息队列中接收消息并打印在终端

send.c

#include "head.h"

struct msgbuf
{
    long mtype;
    char mtext[256];
};

int main(void)
{
    key_t key;
    int msgid = 0;
    int ret = 0;
    ssize_t nret = 0;
    struct msgbuf sendmsg;
    char s[256] = {0};

    key = ftok(".", 's');//第二个参数是个8位非零值
    if (-1 == key)
    {
        perror("fail to ftoksend");
        return -1;
    }
    
    msgid = msgget(key, IPC_CREAT | 0664);//向key对应的IPC对象创建一个消息队列
    if (-1 == msgid)
    {
        perror("fail to msgget");
        return -1;
    }
    while (1)
    {
        memset(&sendmsg, 0, sizeof(sendmsg));
        sendmsg.mtype = 100;//设定要发送的消息的类型
        fgets(sendmsg.mtext, sizeof(sendmsg.mtext), stdin);
        //strcpy(sendmsg.mtext, s);

        ret = msgsnd(msgid, &sendmsg, sizeof(struct msgbuf)-sizeof(long), 0);
        if (ret == -1)
        {
            perror("fail to msgsnd");
            return -1;
        }

        if (!strcmp(sendmsg.mtext, ".quit\n"))
        {
            break;
        }
    }

    msgctl(msgid, IPC_RMID, NULL);

    return 0;
}

recv.c

#include "head.h"

struct msgbuf
{
    long mtype;
    char mtext[256];
};

int main(void)
{
    key_t key;
    int msgid = 0;
    int ret = 0;
    ssize_t nret = 0;
    struct msgbuf recvmsg;
    char s[256] = {0};

    key = ftok(".", 's');//第二个参数是个8位非零值
    if (-1 == key)
    {
        perror("fail to ftoksend");
        return -1;
    }
    
    msgid = msgget(key, IPC_CREAT | 0664);//向key对应的IPC对象创建一个消息队列
    if (-1 == msgid)
    {
        perror("fail to msgget");
        return -1;
    }
    while (1)
    {
        memset(&recvmsg, 0, sizeof(recvmsg));

        nret = msgrcv(msgid, &recvmsg, sizeof(struct msgbuf)-sizeof(long), 100, 0);
        if (ret == -1)
        {
            perror("fail to msgrecv");
            return -1;
        }

        if (!strcmp(recvmsg.mtext, ".quit\n"))
        {
            break;
        }
        printf("RECV: %s", recvmsg.mtext);
    }

    msgctl(msgid, IPC_RMID, NULL);

    return 0;
}

三、共享内存

进程间通信最高效的形式

1、操作方式

       创建共享内存 -> 让两个进程映射到共享内存中 -> 对共享内存做操作 -> 解除映射 -> 删除共享内存

2、函数接口

        1)ftok

        2)shmget

               #include <sys/ipc.h>
               #include <sys/shm.h>

               int shmget(key_t key, size_t size, int shmflg);

                功能:创建一个共享内存

                参数:

                        key:IPC对象的名称

                        size:共享内存的大小

                        shmflg:

                                IPC_CREAT

                                IPC_EXCL

                返回值:成功0,失败-1;

        3)shmat   

               #include <sys/types.h>
               #include <sys/shm.h>

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

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

                参数:

                        shmid:共享内存ID号

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

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

                        shmflg:标志位,0

                返回值:成功返回映射空间的地址,失败返回NULL

        4)shmdt

                int shmdt(const void *shmaddr);

                功能:解除映射

                参数:shmaddr映射的地址

                返回值:成功0,失败-1
        

        5)shmctl

               #include <sys/ipc.h>
               #include <sys/shm.h>

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

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

                参数:

                        shmid:共享内存ID号

                        cmd:IPC_RMID删除共享内存

                        buf:NULL

                返回值:成功返回0,失败返回-1

练习:

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

write.c

#include "head.h"

int main(void)
{
    key_t key;
    int shmid = 0;
    char *p = NULL;

    key = ftok(".", 'a');
    if (key == -1)
    {
        perror("fail to ftokshm");
        return -1;
    }

    shmid = shmget(key, 4096, IPC_CREAT | 0664);//第二个参数为共享内存空间大小
    if (shmid == -1)
    {
        perror("fail to shmget");
        return -1;
    }

    p = (char *)shmat(shmid, NULL, 0);
    if (NULL == p)
    {
        perror("fail to shmat");
        return -1;
    }

    while(1)
    {
        gets(p);
        if (!strcmp(p, ".quit"))
        {
            break;
        }
    }
    
    shmdt(p);
    shmctl(shmid, IPC_RMID, NULL);

    return 0;
}

read.c

#include "head.h"

int main(void)
{
    key_t key;
    int shmid = 0;
    char *p = NULL;
    int cnt = 0;

    key = ftok(".", 'a');
    if (key == -1)
    {
        perror("fail to ftokshm");
        return -1;
    }

    shmid = shmget(key, 4096, IPC_CREAT | 0664);//第二个参数为共享内存空间大小
    if (shmid == -1)
    {
        perror("fail to shmget");
        return -1;
    }

    p = (char *)shmat(shmid, NULL, 0);
    if (NULL == p)
    {
        perror("fail to shmat");
        return -1;
    }

    while(1)
    {
        printf("SHM: %s\n", p);
        if (!strcmp(p, ".quit"))
        {
            break;
        }
    }
    
    shmdt(p);
    shmctl(shmid, IPC_RMID, NULL);

    return 0;
}


http://www.niftyadmin.cn/n/5408753.html

相关文章

未来已来:人工智能时代的创新思维与实践

未来已来&#xff1a;人工智能时代的创新思维与实践 在人工智能时代&#xff0c;创新思维和实践变得尤为重要。以下是对这一主题的小点论述&#xff1a; 1. 理解人工智能的基础 人工智能是一种模拟人类智能的技术&#xff0c;通过机器学习、深度学习等方法来实现。了解人工智…

Kubernetes(K8S第三部分之资源控制器)

资源控制器 什么是控制器 Kubernetes中内建了很多controller&#xff08;控制器&#xff09;&#xff0c;这些相当于一个状态机&#xff0c;用来控制Pod的具体状态和行为。 控制器类型 ReplicationController和ReplicaSet Deployment DaemonSet StateFulSet Job/CronJob…

数据删除

目录 数据删除 删除员工编号为 7369 的员工信息 删除若干个数据 删除公司中工资最高的员工 Oracle从入门到总裁:​​​​​​https://blog.csdn.net/weixin_67859959/article/details/135209645 数据删除 删除数据就是指删除不再需要的数据 delete from 表名称 [where 删…

优优嗨聚集团:美团代运营服务,商家增长的新引擎

在当今数字化时代&#xff0c;线上平台已成为商家拓展业务、提升品牌影响力的重要渠道。美团作为国内领先的本地生活服务平台&#xff0c;拥有庞大的用户群体和丰富的商业资源。然而&#xff0c;对于许多商家而言&#xff0c;如何在美团平台上进行有效运营&#xff0c;实现业务…

Javaweb之SpringBootWeb案例之自动配置的两种常见方案的详细解析

3.2.2.2 方案一 ComponentScan组件扫描 SpringBootApplication ComponentScan({"com.itheima","com.example"}) //指定要扫描的包 public class SpringbootWebConfig2Application {public static void main(String[] args) {SpringApplication.run(Sprin…

sqoop-import 详解

文章目录 前言一、介绍1. sqoop简介2. sqoop import的作用3. 语法3.1 sqoop import 语法3.2 导入配置属性 二、导入参数1. 常见参数2. 验证参数3. 导入控制参数4. 用于覆盖映射的参数5. 增量导入参数6. 输出行格式参数7. 输入解析参数8. Hive 参数9. HBase 参数10. Accumulo 参…

[数据结构]OJ一道------用栈实现队列

题目来源:232. 用栈实现队列 - 力扣&#xff08;LeetCode&#xff09; 解题思路来源:力扣官方题解 https://leetcode.cn/problems/implement-queue-using-stacks/solutions/632369/yong-zhan-shi-xian-dui-lie-by-leetcode-s-xnb6/ 首先我们先来看题目: 给的代码: typedef s…

express+mysql+vue,从零搭建一个商城管理系统9--添加商户

提示&#xff1a;学习express&#xff0c;搭建管理系统 文章目录 前言一、新建models/shop.js二、新建routes/shop.js三、修改routes下的index.js四、添加商户总结 前言 需求&#xff1a;主要学习express&#xff0c;所以先写service部分 一、新建models/shop.js models/shop.…