Linux-进程间通信(IPC)

进程间通信(IPC)介绍

进程间通信(IPC,InterProcess Communication)是指在不同的进程之间传播或交换信息。IPC 的方式包括管道(无名管道和命名管道)、消息队列、信号量、共享内存、Socket、Streams 等。其中 Socket 和 Streams 支持不同主机上的两个进程间通信。

1. 管道

1.1 无名管道

1.1.1 特点
  • 它是半双工的(即数据只能在一个方向上流动),具有固定的读端和写端。
  • 它只能用于父子进程之间的通信。
  • 管道是创建在内存中,进程结束空间释放,管道不复存在。对于它的读写可以使用普通的 readwrite 等函数。
1.1.2 函数原型
#include <unistd.h>
int pipe(int pipefd[2]);
1.1.3 返回值

成功返回 0,失败返回 -1。当一个管道建立时,它会创建两个文件描述符:fd[0] 为读而打开,fd[1] 为写而打开。

1.1.4 无名管道代码示例
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>int main() {int fd[2];int pid;char buf[128];if (pipe(fd) == -1) {printf("create pipe fail\n");}pid = fork();if (pid < 0) {printf("create child fail\n");} else if (pid > 0) {sleep(3);printf("this is father\n");close(fd[0]);write(fd[1], "hello from father", strlen("hello from father"));wait(NULL);} else {printf("this is child\n");close(fd[1]);read(fd[0], buf, 128);printf("read = %s \n", buf);exit(0);}return 0;
}

注意:管道的特性:管道里没有数据时会阻塞。

1.2 命名管道

1.2.1 特点
  • 命名管道可以在无关的进程之间交换数据,与无名管道不同。
  • 命名管道有路径名与之相关联,它以一种特殊设备文件形式存在于文件系统中。
1.2.2 函数原型
#include <sys/types.h>
#include <sys/stat.h>
int mkfifo(const char *pathname, mode_t mode);
1.2.3 命名管道代码示例

read.c

#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>int main() {int fd;char buf[128] = {0};if (mkfifo("./file", 0600) == -1 && errno != EEXIST) {printf("mkfifo fail\n");perror("why");}fd = open("./file", O_RDONLY);printf("read open success\n");while (1) {int n_read = read(fd, buf, 128);printf("read %d byte, context = %s \n", n_read, buf);}close(fd);return 0;
}

write.c

#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>int main() {int fd;char *str = "message from fifo";fd = open("./file", O_WRONLY);printf("write open success\n");while (1) {write(fd, str, strlen(str));sleep(1);}close(fd);return 0;
}

2. 消息队列

消息队列是信息的链接表,存放在内核中。一个消息队列由一个标识符(即队列 ID)来标识。

2.1 特点

  • 消息队列是面向记录的,其中的消息具有特定的格式以及特定的优先级。
  • 消息队列独立于发送与接收进程。进程终止时,消息队列及内容并不会删除。
  • 消息队列可以实现信息的随机查询,消息不一定要以先进先出的次序读取,也可以按消息的类型读取。

2.2 函数原型

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>int msgget(key_t key, int msgflg);
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);
int msgctl(int msqid, int cmd, struct msqid_ds *buf);

2.3 消息队列代码示例

send.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>struct msgbuf {long mtype;char mtext[128];
};int main() {struct msgbuf sendBuf = {888, "this is msg from queue"};struct msgbuf readBuf;key_t key = ftok(".", 1);printf("key=%d\n", key);int msgId = msgget(key, IPC_CREAT | 0777);if (msgId == -1) {perror("why:");}while (1) {msgsnd(msgId, &sendBuf, strlen(sendBuf.mtext), 0);sleep(1);printf("write success\n");msgrcv(msgId, &readBuf, sizeof(readBuf.mtext), 988, 0);sleep(1);printf("read from queue: %s\n", readBuf.mtext);}msgctl(msgId, IPC_RMID, NULL);return 0;
}

read.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>struct msgbuf {long mtype;char mtext[128];
};int main() {struct msgbuf readBuf;struct msgbuf sendBuf = {988, "this is msg from father"};key_t key = ftok(".", 1);printf("key=%d\n", key);int msgId = msgget(key, IPC_CREAT | 0777);if (msgId == -1) {perror("why:");}while (1) {msgrcv(msgId, &readBuf, sizeof(readBuf.mtext), 888, 0);sleep(1);printf("read from queue: %s\n", readBuf.mtext);msgsnd(msgId, &sendBuf, strlen(sendBuf.mtext), 0);sleep(1);printf("write success\n");}msgctl(msgId, IPC_RMID, NULL);return 0;
}

3. 信号

3.1 信号的处理

信号的处理有三种方法:忽略、捕捉和默认动作。具体的信号默认动作可以使用 man 7 signal 来查看系统的具体定义。

3.2 信号处理函数的注册

信号处理函数的注册可以分为入门版和高级版:

  • 入门版:signal 函数
  • 高级版:sigaction 函数

3.3 信号处理发送函数

信号发送函数也分为入门版和高级版:

  • 入门版:kill
  • 高级版:sigqueue

3.4 signal 函数原型及示例

#include <signal.h>typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);

示例

#include <signal.h>
#include <stdio.h>void handler(int signum) {printf("get signum = %d\n", signum);switch (signum) {case SIGINT:printf("SIGINT\n");break;case SIGKILL:printf("SIGKILL\n");break;case SIGUSR1:printf("SIGUSR1\n");break;}
}int main() {signal(SIGINT, handler);signal(SIGKILL, handler);signal(SIGUSR1, handler);while (1);return 0;
}

3.5 kill 函数原型及示例

#include <sys/types.h>
#include <signal.h>
int kill(pid_t pid, int sig);

示例

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <signal.h>int main(int argc, char **argv) {int signum = atoi(argv[1]);int pid = atoi(argv[2]);printf("signum = %d , pid = %d \n", signum, pid);kill(pid, signum);printf("send signal ok\n");return 0;
}

4. 信号量

信号量(semaphore)是一个计数器,用于实现进程间的互斥与同步,而不是用于存储进程间通信数据。

4.1 特点

  • 信号量用于进程间同步,若要在进程间传递数据需要结合共享内存。
  • 信号量基于操作系统的 PV 操作,程序对信号量的操作都是原子操作。
  • 支持信号量组。

4.2 函数原型

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>int semget(key_t key, int nsems, int semflg);
int semop(int semid, struct sembuf *sops, unsigned nsops);
int semctl(int semid, int semnum, int cmd, ...);

4.3 信号量代码示例

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <stdio.h>//联合体,用于 semctl 初始化
union semun {int val;struct semid_ds *buf;unsigned short *array;
};void pGetKey(int id) {struct sembuf set;set.sem_num = 0;set.sem_op = -1;set.sem_flg = SEM_UNDO;semop(id, &set, 1);printf("get key\n");
}void pPutBackKey(int id) {struct sembuf set;set.sem_num = 0;set.sem_op = 1;set.sem_flg = SEM_UNDO;semop(id, &set, 1);printf("put back the key\n");
}int main() {int semid;key_t key = ftok(".", 2);//1. 获取或创建信号量semid = semget(key, 1, IPC_CREAT | 0666);union semun initsem;initsem.val = 0;//2. 初始化信号量semctl(semid, 0, SETVAL, initsem);int pid = fork();if (pid > 0) {//4. 拿锁pGetKey(semid);printf("this is father\n");//5. 还锁pPutBackKey(semid);//6. 销毁锁semctl(semid, 0, IPC_RMID);} else if (pid == 0) {printf("this is child\n");//3. 放锁pPutBackKey(semid);} else {printf("fork error\n");}return 0;
}

5. 共享内存

共享内存(shared memory)指两个或多个进程共享一个给定的存储区。

5.1 特点

  • 共享内存是最快的一种 IPC,因为进程是直接对内存进行存储,而不需要任何数据的拷贝。
  • 只能单独一个进程写或读,如果 A 和 B 进程同时写,会造成数据的混乱(需要搭配信号量来使用)。

5.2 函数原型

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>int shmget(key_t key, size_t size, int shmflg);
void *shmat(int shmid, const void *shmaddr, int shmflg);
int shmdt(const void *shmaddr);
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

5.3 共享内存代码示例

shm_write.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>int main() {int shmid;char *shmaddr;key_t key = ftok(".", 1);// 创建共享内存shmid = shmget(key, 1024 * 4, IPC_CREAT | 0600);if (shmid == -1) {printf("create shm fail\n");exit(-1);}// 连接映射共享内存shmaddr = shmat(shmid, 0, 0);printf("shmat OK\n");// 将数据拷贝到共享内存strcpy(shmaddr, "hello world\n");sleep(5); // 等待 5 秒,避免一下子断开连接。等待另外一个进程读完。// 断开共享内存连接shmdt(shmaddr);// 删除共享内存shmctl(shmid, IPC_RMID, 0);printf("quit\n");return 0;
}

shm_get.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>int main() {int shmid;char *shmaddr;key_t key = ftok(".", 1);// 打开共享内存shmid = shmget(key, 1024 * 4, 0);if (shmid == -1) {printf("create shm fail\n");exit(-1);}// 连接并映射共享内存shmaddr = shmat(shmid, 0, 0);printf("get from shm_write message is: %s", shmaddr);// 断开共享内存连接shmdt(shmaddr);printf("quit\n");return 0;
}

6. 结合消息队列、共享内存、信号量的示例

6.1 代码示例

get.c

#include <stdio.h>
#include <stdlib.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <sys/msg.h>
#include <string.h>// 消息队列结构
struct msg_form {long mtype;char mtext;
};// 联合体,用于 semctl 初始化
union semun {int val;struct semid_ds *buf;unsigned short *array;
};// 初始化信号量
int init_sem(int sem_id, int value) {union semun tmp;tmp.val = value;if (semctl(sem_id, 0, SETVAL, tmp) == -1) {perror("Init Semaphore Error");return -1;}return 0;
}// P 操作
int sem_p(int sem_id) {struct sembuf sbuf;sbuf.sem_num = 0;sbuf.sem_op = -1;sbuf.sem_flg = SEM_UNDO;if (semop(sem_id, &sbuf, 1) == -1) {perror("P operation Error");return -1;}return 0;
}// V 操作
int sem_v(int sem_id) {struct sembuf sbuf;sbuf.sem_num = 0;sbuf.sem_op = 1;sbuf.sem_flg = SEM_UNDO;if (semop(sem_id, &sbuf, 1) == -1) {perror("V operation Error");return -1;}return 0;
}// 删除信号量集
int del_sem(int sem_id) {union semun tmp;if (semctl(sem_id, 0, IPC_RMID, tmp) == -1) {perror("Delete Semaphore Error");return -1;}return 0;
}// 创建一个信号量集
int create_sem(key_t key) {int sem_id;if ((sem_id = semget(key, 1, IPC_CREAT | 0666)) == -1) {perror("semget error");exit(-1);}init_sem(sem_id, 1);  // 初值设为 1 资源未占用return sem_id;
}int main() {key_t key;int shmid, semid, msqid;char *shm;struct shmid_ds buf1;  // 用于删除共享内存struct msqid_ds buf2;  // 用于删除消息队列struct msg_form msg;   // 消息队列用于通知对方更新了共享内存// 获取 key 值if ((key = ftok(".", 'z')) < 0) {perror("ftok error");exit(1);}// 创建共享内存if ((shmid = shmget(key, 1024, IPC_CREAT | 0666)) == -1) {perror("Create Shared Memory Error");exit(1);}// 连接共享内存shm = (char *)shmat(shmid, 0, 0);if ((int)shm == -1) {perror("Attach Shared Memory Error");exit(1);}// 创建消息队列if ((msqid = msgget(key, IPC_CREAT | 0777)) == -1) {perror("msgget error");exit(1);}// 创建信号量semid = create_sem(key);// 读数据while (1) {msgrcv(msqid, &msg, 1, 888, 0);  // 读取类型为 888 的消息if (msg.mtext == 'q')  // quit - 跳出循环break;if (msg.mtext == 'r')  // read - 读共享内存{sem_p(semid);printf("%s\n", shm);sem_v(semid);}}// 断开连接shmdt(shm);// 删除共享内存、消息队列、信号量shmctl(shmid, IPC_RMID, &buf1);msgctl(msqid, IPC_RMID, &buf2);del_sem(semid);return 0;
}

send.c

#include <stdio.h>
#include <stdlib.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <sys/msg.h>
#include <string.h>// 消息队列结构
struct msg_form {long mtype;char mtext;
};// 联合体,用于 semctl 初始化
union semun {int val;struct semid_ds *buf;unsigned short *array;
};// P 操作
int sem_p(int sem_id) {struct sembuf sbuf;sbuf.sem_num = 0;sbuf.sem_op = -1;sbuf.sem_flg = SEM_UNDO;if (semop(sem_id, &sbuf, 1) == -1) {perror("P operation Error");return -1;}return 0;
}// V 操作
int sem_v(int sem_id) {struct sembuf sbuf;sbuf.sem_num = 0;sbuf.sem_op = 1;sbuf.sem_flg = SEM_UNDO;if (semop(sem_id, &sbuf, 1) == -1) {perror("V operation Error");return -1;}return 0;
}int main() {key_t key;int shmid, semid, msqid;char *shm;struct msg_form msg;int flag = 1;  // while 循环条件// 获取 key 值if ((key = ftok(".", 'z')) < 0) {perror("ftok error");exit(1);}// 获取共享内存if ((shmid = shmget(key, 1024, 0)) == -1) {perror("shmget error");exit(1);}// 连接共享内存shm = (char *)shmat(shmid, 0, 0);if ((int)shm == -1) {perror("Attach Shared Memory Error");exit(1);}// 创建消息队列if ((msqid = msgget(key, 0)) == -1) {perror("msgget error");exit(1);}// 获取信号量if ((semid = semget(key, 0, 0)) == -1) {perror("semget error");exit(1);}// 写数据printf("***************************************\n");printf("*                 IPC                 *\n");printf("*    Input r to send data to server.  *\n");printf("*    Input q to quit.                 *\n");printf("***************************************\n");while (flag) {char c;printf("Please input command: ");scanf("%c", &c);switch (c) {case 'r':printf("Data to send: ");sem_p(semid);  // 访问资源scanf("%s", shm);sem_v(semid);  // 释放资源// 清空标准输入缓冲区while ((c = getchar()) != '\n' && c != EOF);msg.mtype = 888;msg.mtext = 'r';  // 发送消息通知服务器读数据msgsnd(msqid, &msg, sizeof(msg.mtext), 0);break;case 'q':msg.mtype = 888;msg.mtext = 'q';msgsnd(msqid, &msg, sizeof(msg.mtext), 0);flag = 0;break;default:printf("Wrong input!\n");// 清空标准输入缓冲区while ((c = getchar()) != '\n' && c != EOF);}}// 断开连接shmdt(shm);return 0;
}

7. 对比总结

通过上述对比可以看出,各种 IPC 方式各有优劣,选择合适的方式进行进程间通信可以提高程序的效率和可靠性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/36878.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

纠正自己的做题习惯-认真学习理科第三天

自己的做题习惯有问题&#xff0c;自己得不断地纠正自己的做题习惯&#xff0c;自己经常题目都不看&#xff0c;就开始写题&#xff0c;结果就是自己没有办法能够良好地进行做题。 any()函数用于判断数组中是否存在满足条件的元素&#xff0c;而all&#xff08;&#xff09;函数…

LabVIEW编程控制ABB机械臂

使用LabVIEW编程控制ABB机械臂是一项复杂但十分有价值的任务。通过LabVIEW&#xff0c;可以实现对机械臂的精确控制和监控&#xff0c;提升自动化水平和操作效率。 1. 项目规划和硬件选型 1.1 确定系统需求 运动控制&#xff1a;确定机械臂需要执行的任务&#xff0c;如抓取、…

typescript定义函数的传参、返回值

Render 函数中定义函数传参 interface List {id: number;name: string; }interface Result {data: List[]; //表示由 List 接口组成的数组 }function Render(result: Result) {result.data.forEach(value > {console.log(value);}); }let result {data: [{id: 1,name: 张三…

嵌入式Linux系统编程 — 5.2 Linux系统时间与日期

目录 1 了解Linux系统时间 1.1 几种常用的时间 1.2 如何查看几种常用的时间 1.3 Linux 系统中的时间 2 time、gettimeofday获取时间 2.1 time函数 2.2 ​​​​​​​gettimeofday函数&#xff1a; 2.3 示例程序 3 时间转换函数 3.1 ctime与ctime_r函数 3.2 localti…

Unity之自定义Text组件默认属性值

内容将会持续更新&#xff0c;有错误的地方欢迎指正&#xff0c;谢谢! Unity之自定义Text组件默认属性值 TechX 坚持将创新的科技带给世界&#xff01; 拥有更好的学习体验 —— 不断努力&#xff0c;不断进步&#xff0c;不断探索 TechX —— 心探索、心进取&#xff01;…

普通集群与镜像集群配置

一. 环境准备 关闭防火墙和selinux&#xff0c;进行时间同步 rabbitmq-1 Rocky_linux9.4 192.168.226.22rabbitmq-2Rocky_linux9.4192.168.226.23rabbitmq-3Rocky_linux9.4192.168.226.24 修改主机名#192.168.226.22 hostnamectl set-hostname rabbitmq-1#192.168.226.22 ho…

isalpha()方法——判断字符串是否只由字母组成

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 语法参考 isalpha()方法用于判断字符串是否只由字母组成。isalpha()方法的格式如下&#xff1a; str.isalpha() 如果字符串中至少有一个字符并且所…

「C++系列」C++ 基本语法

文章目录 一、C 基本语法1. **预处理指令**2. **数据类型**3. **变量和常量**4. **运算符**5. **控制流语句**6. **函数**7. **数组**8. **指针和引用**9. **结构体和类**10. **输入/输出**11. **异常处理**12. **命名空间** 二、C 程序结构1. 程序流程结构2. 文件结构3. 编译预…

银河麒麟高级服务器操作系统V10SP2(X86)配置bond0的mac地址为指定子网卡的mac地址

银河麒麟高级服务器操作系统V10SP2&#xff08;X86&#xff09;配置bond0的mac地址为指定子网卡的mac地址 一 系统环境二 删除和备份原有配置2.1 down掉bond02.2 备份之前的bond配置到/root/bak2.3 删除bond配置&#xff08;网卡文件根据实际情况变化&#xff09; 三 新建bond0…

昇思25天学习打卡营第6天|MindSpore-ResNet50迁移学习

MindSpore-ResNet50迁移学习 迁移学习 图像的迁移学习是一种机器学习方法,它将已经在一个或多个源任务上训练好的预训练模型应用到新的目标任务上。这种方法的核心思想是,通过在大量数据上训练模型,学习到的特征表示可以用于其他相关任务,从而提高目标任务的性能。 应用场…

MySQL之聚簇索引和非聚簇索引

1、什么是聚簇索引和非聚簇索引&#xff1f; 聚簇索引&#xff0c;通常也叫聚集索引。 非聚簇索引&#xff0c;指的是二级索引。 下面看一下它们的含义&#xff1a; 1.1、聚集索引选取规则 如果存在主键&#xff0c;主键索引就是聚集索引。如果不存在主键&#xff0c;将使…

【Lua】第二篇:打印函数和注释

文章目录 一. 打印函数二. 注释方式1. 单行注释2. 多行注释 一. 打印函数 Lua 程序是以 .lua 结尾的文件&#xff0c;创建一个的 Test.lua 的文件&#xff0c;使用 print 函数输出字符串"Hello World"&#xff1a; print(Hello World) 保存之后使用命令lua 文件名编…

MyBatis(14)MyBatis 如何实现多对一和一对多的映射关系

MyBatis 中实现多对一和一对多的映射主要依赖于映射文件中的 <association> 和 <collection> 标签。这两种映射关系的实现&#xff0c;是通过嵌套查询或嵌套结果映射来完成的。在源码层面&#xff0c;MyBatis 使用相应的处理器来处理这些标签&#xff0c;最终实现复…

elasticsearch-6.8.23的集群搭建过程

三个节点的 ElasticSearch 集群搭建步骤 准备三台机器&#xff1a;28.104.87.98、28.104.87.100、28.104.87.101 和 ElasticSearch 的安装包 elasticsearch-6.8.23.tar.gz ----------------------------- 28.104.87.98&#xff0c;使用 root 用户操作 ----------------------…

Java | Leetcode Java题解之第206题反转链表

题目&#xff1a; 题解&#xff1a; class Solution {public ListNode reverseList(ListNode head) {if (head null || head.next null) {return head;}ListNode newHead reverseList(head.next);head.next.next head;head.next null;return newHead;} }

SpringBoot学习04-[定制SpringMVC]

定制SpringMVC 定制SpringMvc的自动配置定制springmvc-configurePathMatch配置定制SpringMVC-拦截器Interceptor定制SpringMVC-CORS配置全局cors配置针对某个方法加跨域解决 WebMvcConfigurer原理定制SpringMVC-JSONJSON开发jackson的使用定制化json序列化和反序列化 JSON国际化…

CriticGPT: 用 GPT-4 找出 GPT-4 的错误

CriticGPT 是 OpenAI 发布的一个基于 GPT-4 的模型&#xff0c;它可以帮助我们人类 Review 并纠正 ChatGPT 在生成代码时的错误。使用 CriticGPT 审查代码时&#xff0c;有 60% 的概率生成的代码更好更正确。

最近写javaweb出现的一个小bug---前端利用 form 表单传多项数据,后端 Servlet 取出的各项数据均为空

目录&#xff1a; 一. 问题引入二 解决问题 一. 问题引入 近在写一个 java web 项目时&#xff0c;遇到一个让我头疼了晚上的问题&#xff1a;前端通过 post 提交的 form 表单数据可以传到后端&#xff0c;但当我从 Servlet 中通过 request.getParameter(“name”) 拿取各项数…

【小沐学AI】Python实现语音识别(faster-whisper-webui)

文章目录 1、简介1.1 whisper1.2 faster-whisper 2、安装3、测试结语 1、简介 1.1 whisper https://github.com/openai/whisper Whisper 是一种通用语音识别模型。它是在各种音频的大型数据集上训练的&#xff0c;也是一个多任务模型&#xff0c;可以执行多语言语音识别、语音…

软考初级网络管理员_04_硬件单选题

1.进程是程序的一次执行&#xff0c;()是进程存在的唯一标志。 程序 数据 进程控制块 进程标识符 2.(请作答此空)是计算机内部运算部件一次能同时处理的二进制数据位数。计算机的运行速度使用每秒钟所能执行的()指令数目来评估。 字长 带宽 内存 寄存器 3.某银行的一…