system v IPC进程通信 5.17

本文介绍了SystemVIPC中的三种主要通信机制:共享内存、消息队列和信号灯集。通过示例代码展示了如何使用shmget创建和操作共享内存,msgget和msgsnd/msgrcv用于消息队列的发送和接收,以及semget和semop用于信号灯集的控制。这些机制允许不同进程间高效地交换数据。
摘要由CSDN通过智能技术生成

进程间通信

system v IPC对象

共享内存,消息队列,信号灯集

  1. 相关操作命令:
1.查询系统所有ipc对象:ipcs

2.查询系统共享内存:ipcs -m

3.查询系统消息队列:ipcs -q

4.查询系统信号灯集:ipcs -s

5.删除系统ipc对象:iprm [-m -q -s] ID

共享内存

  1. 共享内存:
1.是进程间通信,效率最高的一种方式
2.内核专门留出的一块内存区域,要使用的用户进程 直接映射到自己的地址空间
3.用户进程就能直接对这块内存区域进程读写操作,而不需要进行数据拷贝,效率就会提高
共享内存操作步骤
1.创建或打开共享内存
2.映射:将共享内存首地址 映射到进程地址空间,方便访问使用
3.解除映射
4.删除共享内存

shmget()

所需头文件
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
函数原型
int shmget(key_ t key, int size, int shmflg);
函数参数
key: IPC_ PRIVATE或ftok的返回值
size:共享内存区大小
shmflg:同open函 数的权限位,也可以用8进制表示法
函数返回值
成功:共享内存段标识符
出错: -1

ftok()

所需头文件
#include <sys/types.h>

#include <sys/ipc.h>

函数原型
key_ t ftok(const char *pathname, int proj_ id);
函数参数
const char *pathname:文件路径名
int proj_ id: 子序号
函数返回值
成功:返回生成的key值
出错: -1

shmat()

所需头文件
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
函数原型
void * shmat(int shmid, const void *shmaddr; int shmflg);
函数参数
shmid:要映射的共享内存区标识符
shmaddr:将共享内存映射到指定地址(若为NULL,则表示由系统自动完成映射)
shmflg :
SHM_ RDONLY:共享内存只读
默认0:共享内存可读写
函数返回值
成功:映射后的地址
出错: -1

shmdt()

所需头文件
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
函数原型
int shmdt(const void *shmaddr);
函数参数
shmaddr:共享内存映射后的地址
函数返回值
成功: 0
出错: -1

shmctl()

所需头文件
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
函数原型
int shmctl(int shmid, int cmd, struct shmid_ ds *buf);
函数参数
shmid:要操作的共享内存标识符
cmd: IPC_ STAT (获取对象属性)
IPC_ SET (设置对象属性)
IPC_ RMID (删除对象)
buf:指定IPC_ STAT/IPC _SET时用以保存/设置属性
函数返回值
成功: 0
出错: -1

/*===============================================
*   文件名称:shareMemory.c
*   创 建 者:memories 
*   创建日期:2023年05月17日
*   描    述:
================================================*/


在有亲缘关系的父子进程之间的shareMemory通信


#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char *argv[])
{ 
    //1.创建或者打开共享内存
   int shmid =  shmget(IPC_PRIVATE,64,0600);
    if(shmid<0)
    {
        perror("shmget");
        return -1;
    }
    printf("shmid=%d\n",shmid);
    system("ipcs -m");//执行字符串的命令,属于系统调用
    //2.映射共享内存首地址到用户进程空间
    char *p = shmat(shmid,NULL,0);
    if(p==(char *)-1)
    {
        perror("shmat");
        goto end2;
    }

    pid_t pid=fork();
    if(pid<0)
    {
        perror("fork");
        goto end1;
    }
    else if(pid==0)
    {
        while(1)
        {
            fgets(p,32,stdin);
            if(strncmp(p,"quit",4)==0)
                break;
        }
        exit(0);
    }
    else
    {
        while(1)
        {
            printf("p=%s\n",p);
            if(strncmp(p,"quit",4)==0)
                break;
            sleep(3);
        }
end1:
    //3.解除映射
    if(shmdt(p)<0)
    {
        perror("shmdt");
    }
    printf("删除映射成功\n");


end2:
    //4.删除共享内存
    if(shmctl(shmid,IPC_RMID,NULL)<0)
    {
        perror("shmctl");
        char cmd[32]={0};
        sprintf(cmd,"ipcrm -m %d",shmid);
        system(cmd);
    }
    printf("删除共享内存成功\n");
    system("ipcs -m");
    }
    return 0;
} 

/*===============================================
*   文件名称:shareMemory_read.c
*   创 建 者:memories 
*   创建日期:2023年05月17日
*   描    述:
================================================*/

在不同进程非亲缘关系间的进程shareMemory通信

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char *argv[])
{ 
    key_t key=ftok(".",3);
    if(key==-1)
    {
        perror("key");
        return -1;
    }
    printf("key=%d\n",key);
    //1.创建或者打开共享内存
   int shmid =  shmget(key,64,0600|IPC_CREAT);
    if(shmid<0)
    {
        perror("shmget");
        return -1;
    }
    printf("shmid=%d\n",shmid);
    system("ipcs -m");//执行字符串的命令,属于系统调用
    //2.映射共享内存首地址到用户进程空间
    char *p = shmat(shmid,NULL,0);
    if(p==(char *)-1)
    {
        perror("shmat");
        goto end2;
    }

        while(1)
        {
            printf("p=%s\n",p);
            if(strncmp(p,"quit",4)==0)
                break;
            sleep(3);
        }

end1:
    //3.解除映射
    if(shmdt(p)<0)
    {
        perror("shmdt");
    }
    printf("删除映射成功\n");


end2:
    //4.删除共享内存
    if(shmctl(shmid,IPC_RMID,NULL)<0)
    {
        perror("shmctl");
        char cmd[32]={0};
        sprintf(cmd,"ipcrm -m %d",shmid);
        system(cmd);
    }
    printf("删除共享内存成功\n");
    system("ipcs -m");
    
    return 0;
} 


/*===============================================
*   文件名称:shareMemory_write.c
*   创 建 者:memories 
*   创建日期:2023年05月17日
*   描    述:
================================================*/
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char *argv[])
{ 
    //根据文件名或路径,子序号生成key
    key_t key=ftok(".",3);
    if(key==-1)
    {
        perror("key");
        return -1;
    }
    printf("key=%d\n",key);
    //1.创建或者打开共享内存
   int shmid =  shmget(key,64,0600|IPC_CREAT);
    if(shmid<0)
    {
        perror("shmget");
        return -1;
    }
    printf("shmid=%d\n",shmid);
    system("ipcs -m");//执行字符串的命令,属于系统调用
    //2.映射共享内存首地址到用户进程空间
    char *p = shmat(shmid,NULL,0);
    if(p==(char *)-1)
    {
        perror("shmat");
        goto end2;
    }

        while(1)
        {
            fgets(p,32,stdin);
            if(strncmp(p,"quit",4)==0)
                break;
        }
        exit(0);
end1:
    //3.解除映射
    if(shmdt(p)<0)
    {
        perror("shmdt");
    }
    printf("删除映射成功\n");


end2:
    //4.删除共享内存
    if(shmctl(shmid,IPC_RMID,NULL)<0)
    {
        perror("shmctl");
        char cmd[32]={0};
        sprintf(cmd,"ipcrm -m %d",shmid);
        system(cmd);
    }
    printf("删除共享内存成功\n");
    system("ipcs -m");
    
    return 0;
} 


消息队列

​ 在内核中创建的队列,该队列由 消息队列ID来唯一标识。
​ 消息队列就是一个消息的列表,用户可以 添加消息、读取消息。
​ 消息队列 必须按照消息的类型 来进行 添加和读取。

msgget()

​ 1、打开或者创建信息队列
​ int msgget(key_t key, int msgflg);
​ 头文件:
​ #include <sys/types.h>
​ #include <sys/ipc.h>
​ #include <sys/msg.h>
​ 参数:
​ key :同 shmget函数的key值一样
​ msgflg :消息队列的权限,同 open一样
​ 返回值:
​ 成功:消息队列的 msgid
​ 失败:-1

msgsnd()

​ 2、添加消息
​ int msgsnd(int msqid, const void *msgp, size_t size, int flag);
​ 参数:
​ msqid :消息队列的 ID
​ msgp :存储消息的结构体首地址
​ size :添加的本条消息的正文长度
​ flag :0阻塞等待添加成功才返回, IPC_NOWAIT:非阻塞
​ 返回值:
​ 成功:0
​ 失败:-1
​ 注意:
​ 消息结构体需要用户自行定义
​ struct msgbuf{
​ long type; //存储当前消息的类型
​ char text[N]; //存储当前消息的正文
​ };
​ ***每发送一条消息,都必须指定该条消息的类型

msgrcv()

3、读取消息
int msgrcv(int msgid, void * msgp, size_t size, long msgtype, int flag);
参数:
msqid :消息队列的 ID
msgp :存储消息的结构体首地址
size :要读取的消息正文长度
msgtype :要读取的消息的类型
flag :0阻塞,IPC_NOWAIT:非阻塞
返回值:
成功:接受的消息的长度
失败:-1
注意:
消息结构体需要用户自行定义
struct msgbuf{
long type; //存储当前消息的类型
char text[N]; //存储当前消息的正文
};
要读取一条消息时,必须指定接收哪种类型的消息

/*===============================================
*   文件名称:messageQueue_read.c
*   创 建 者:memories 
*   创建日期:2023年05月17日
*   描    述:
================================================*/


在不同进程非亲缘关系间的进程messagequeue通信

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>
#include <string.h>

#define  N 64
typedef struct{
    long mtype;
    char mtex[N]; 
}msgbuf;

int main(int argc, char *argv[])
{ 
    //创建key值
    key_t key=ftok(".",3);
    if(key==-1)
    {
        perror("key");
        return -1;
    }
    printf("key=%d\n",key);
    //创建消息队列
    int msqid=msgget(key,0600|IPC_CREAT);
    if(msqid<0)
    {
        perror("msgget");
        return -1;
    }
    printf("msqid=%d\n",msqid);
    system("ipcs -q");
    //发送消息
        while(1)
        {
            //定义结构体变量
            msgbuf msg;
            msgrcv(msqid,&msg,sizeof(msg),1,0);
            printf("rcv=%s\n",msg.mtex);
            if(strncmp(msg.mtex,"quit",4)==0)
                break;
            sleep(3);
        }

        //删除消息队列
        if((msgctl(msqid,IPC_RMID,NULL))<0)
        {
            perror("msgctl");
        }
        system("ipcs -q");
    return 0;
} 


/*===============================================
*   文件名称:messageQueue_write.c
*   创 建 者:memories 
*   创建日期:2023年05月17日
*   描    述:
================================================*/
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>
#include <string.h>

#define  N 64
typedef struct{
    long mtype;
    char mtex[N]; 
}msgbuf;

int main(int argc, char *argv[])
{ 
    //创建key值
    key_t key=ftok(".",3);
    if(key==-1)
    {
        perror("key");
        return -1;
    }
    printf("key=%d\n",key);
    //创建消息队列
    int msqid=msgget(key,0600|IPC_CREAT);
    if(msqid<0)
    {
        perror("msgget");
        return -1;
    }
    printf("msqid=%d\n",msqid);
    system("ipcs -q");
    //发送消息
        while(1)
        {
            //定义结构体变量
            msgbuf msg;
            //设置本条消息的类型
            msg.mtype = 1;
            //从标准输入流中获取一行前32字节到msg结构体的mtex中
            fgets(msg.mtex,32,stdin);
            //将消息添加到消息队列中
            msgsnd(msqid,&msg,sizeof(msg)-sizeof(long),0);

            if(strncmp(msg.mtex,"quit",4)==0)
                break;
            sleep(3);
        }

        //删除消息队列
        if((msgctl(msqid,IPC_RMID,NULL))<0)
        {
            perror("msgctl");
        }
        system("ipcs -q");
    return 0;
} 


信号灯集

​ 信号灯的种类:

	有名信号灯
	无名信号灯(基于内存的信号灯)
	 system v IPC 信号灯集

信号灯:分为两类
二值信号灯:信号量的数值只能是 0或者1,类似于互斥
计数信号灯:信号量的数值在 0到 n

semget()

1、创建或者打开信号灯集
int semget(key_t key, int nsems, int semflg);
功能:
打开或者创建打开信号灯集,设置集合中信号灯的个数和权限
头文件:
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
参数:
key :同 shmget函数一致
nsems :设置集合中信号灯的 数量
semflg :同open函数一样,权限8进制和 IPC_CREAT
返回值:
成功:信号灯集的ID号
失败:-1
注意:创建信号灯集后,必须要先行设置 集合中每隔信号灯的数值
union semun {
int val; /* Value for SETVAL */
struct semid_ds buf; / Buffer for IPC_STAT, IPC_SET */
unsigned short array; / Array for GETALL, SETALL */
struct seminfo __buf; / Buffer for IPC_INFO
(Linux-specific) */
};

函数:
int semctl(int semid, int semnum, int cmd, union semnu val);
功能:
设置灯集中某个编号的信号灯的数值
参数:
semid: 信号灯集的ID号
semnu: 设置的信号灯的编号
cmd : SETVAL 设置数值
val : 联合体变量,存储要设置的信号灯数值
返回值:
成功:0
失败:-1

semop()

2、信号灯集的操作
int semop( int semid, struct sembuf *opsptr, size_t nops);
功能:
设置灯集中某个信号灯的 操作为 p还是v
参数:
semid :信号灯集的ID号
opsptr : 结构体首地址
nops :操作的信号灯个数
返回值:
成功:0
失败:-1
struct sembuf {
short sem_num; // 要操作的信号灯的编号
short sem_op; // 0 : 等待,直到信号灯的值变成0
// 1 : 释放资源,V操作
// -1 : 分配资源,P操作
short sem_flg; // 0, IPC_NOWAIT, SEM_UNDO
};

semctl()

3、信号灯集的控制函数
int semctl ( int semid, int semnum, int cmd…/*union semun arg */);
参数:
semid :信号灯集ID
semnum : 要修改的信号灯编号
cmd : GETVAL:获取信号灯的值 SETVAL:设置信号灯的值 IPC_RMID:从系统中删除信号灯集合
返回值:
成功:0
失败:-1

/*===============================================
 *   文件名称:semaphore.c
 *   创 建 者:memories 
 *   创建日期:2023年05月17日
 *   描    述:
 ================================================*/


利用semaphore 实现子父进程之间的通信,当子进程输入数据时,父进程等待,父进程输出数据到终端时,子进程阻塞等待


#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>
#include <string.h>
#include <sys/sem.h>


//使用semctl必须在程序中定义联合体
union semun {
    int val;    /* Value for SETVAL */
    struct semid_ds *buf;    /* Buffer for IPC_STAT, IPC_SET */
    unsigned short  *array;  /* Array for GETALL, SETALL */
    struct seminfo  *__buf;  /* Buffer for IPC_INFO
                                (Linux-specific) */
};



int main(int argc, char *argv[])
{ 
    //1.创建或者打开共享内存
    int shmid =  shmget(IPC_PRIVATE,64,0600);
    if(shmid<0)
    {
        perror("shmget");
        return -1;
    }
    printf("shmid=%d\n",shmid);
    system("ipcs -m");//执行字符串的命令,属于系统调用
    //2.映射共享内存首地址到用户进程空间
    char *p = shmat(shmid,NULL,0);
    if(p==(char *)-1)
    {
        perror("shmat");
        goto end2;
    }
    //创建或打开信号灯集,设置灯集中信号灯的个数,以及集合的权限
    int semid = semget(IPC_PRIVATE,2,IPC_CREAT|0600);
    if(semid < 0)
    {
        perror("semget");
        return -1;
    }

    //设置灯集中,每个信号灯的初始值(初始化灯集)
    //定义联合体变量,给成员val赋值为1
    union semun val1 ={1};
    //设置灯集中编号 0 的信号灯的数值为val1
    semctl(semid,0,SETVAL,val1);

    union semun val2 = {0};
    //设置灯集中编号为1的信号灯的数值为val2
    semctl(semid,1,SETVAL,val2);

    pid_t pid=fork();
    if(pid<0)
    {
        perror("fork");
        goto end1;
    }
    else if(pid==0)
    {
        while(1)
        {
            //p(0)
            struct sembuf buf = {
                .sem_num = 0,//指定要操作的信号灯标号
                .sem_op = 1,//设置要执行的操作 p操作(1)
                .sem_flg = 0//设置当前操作为阻塞模式
            };
            semop(semid,&buf,1);//进行p操作
            fgets(p,32,stdin);
            //v(1)
            buf.sem_num = 1;//设置要操作的信号灯编号
            buf.sem_op = -1;//设置要执行的操作v操作(-1)
            buf.sem_flg = 0;//设置当前操作为阻塞模式
            semop(semid,&buf,1);//进行v 操作
            if(strncmp(p,"quit",4)==0)
                break;
        }
        exit(0);
    }
    else
    {
        while(1)
        {
            //p(1)
            struct sembuf buf = {
                .sem_num = 1,//指定要操作的信号灯标号
                .sem_op = 1,//设置要执行的操作 p操作(1)
                .sem_flg = 0//设置当前操作为阻塞模式
            };
            semop(semid,&buf,1);//进行p操作
            printf("%s\n",p);
            //v(0)
            buf.sem_num = 0;//设置要操作的信号灯编号
            buf.sem_op = -1;//设置要执行的操作v操作(-1)
            buf.sem_flg = 0;//设置当前操作为阻塞模式
            semop(semid,&buf,1);//进行v 操作
            if(strncmp(p,"quit",4)==0)
                break;
        }
    }
end1:
        //3.解除映射
        if(shmdt(p)<0)
        {
            perror("shmdt");
        }
        printf("删除映射成功\n");


end2:
        //4.删除共享内存
        if(shmctl(shmid,IPC_RMID,NULL)<0)
        {
            perror("shmctl");
            char cmd[32]={0};
            sprintf(cmd,"ipcrm -m %d",shmid);
            system(cmd);
        }
        printf("删除共享内存成功\n");
        system("ipcs -m");
    
    return 0;
} 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

孤独memories

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值