当前位置: 首页 > news >正文

进程间通信(重点)

概念 

  • 进程是一个独立的资源分配单元,不同进程之间的资源是独立的
  • 进程并非孤立的,不同进程需要进行信息的交互和状态的传递,因此需要进程之间的通信【IPC: Inter processes communication】

如qq聊天,qq在每个人的手机上是独立的,聊天室不是孤立的

进程间通信的目的:

  • 数据传输:一个进程需要将它的数据发送给另一个进程
  • 通知事件:一个进程需要向另一个或一组进程发送消息,通知他们发生了事件
  • 资源共享:多个进程之间共享同样的资源,需要内核提供互斥和同步机制

同步和异步:

  • 进程控制:有些进程希望完全控制另一进程的执行

管道:

什么是进程间通信


        Linux环境下,进程地址空间相互独立,每个进程各自有不同的用户地址空间。任何一个进程的全局变量在另一个进程中都看不到,所以进程和进程之间不能相互访问,要交换数据必须通过内核,在内核中开辟一块缓冲区,进程1把数据从用户空间拷到内核缓冲区,进程2再从内核缓冲区把数据读走,内核提供的这种机制称为进程间通信(IPC,InterProcess Communication)。

 

 管道的特点:

  1. 其实是一个在内核内存中维护的缓冲区
  2. 管道拥有文件的特质:读操作,写操作。匿名管道,有名管道。按照操作文件的方式对管道进行操作
  3. 一个管道是一个字节流,bytes
  4. 管道的传递数据是顺序的,单向的。一端写,一端读,半双工的,同一时间只能向一个方向。如对讲机,读取是单向的,但是可以双向传输。
  5. 使用特性

 为什么可以进行管道间的通信?

子进程被fork()出来后,子进程和父进程共享文件描述符

  管道数据结构:

循环队列:利用资源

管道的使用:

 管道创建步骤:(重点)

  • 父进程调用pipe函数创建管道,得到两个文件描述符fd[0]和fd[1],分别指向管道的读端和写端。

父进程调用fork创建子进程,那么子进程也有两个文件描述符指向同一管

 

父进程关闭管道读端,子进程关闭管道写端。父进程可以向管道中写入数据,子进程将管道中的数据读出,这样就实现了父子进程间通信

 

参考:使用管道完成父子进程间通信_执迷C++ 的菜鸡的博客-CSDN博客_创建一个有名管道父进程负责将指定的文件内容通过管道发给子进程子进程负责将 很简洁很棒!!看图很容易理解

使用

/*  
man pipe
#include <unistd.h>
int pipe(int pipefds[2]);创建一个匿名管道,用来进程间的通信
*/
#include <unistd.h>
#include <sys/types.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main(){//创建管道,在fork()之前int pipefds[2];     //得到两个文件描述符fd[0]和fd[1],分别指向管道的读端和写端。int ret = pipe(pipefds);if (ret == -1){perror("pipe");exit(0);}// 创建子进程pid_t pid = fork();if (pid > 0){printf("i am parent process,pid: %d\n",getpid());char buf[1024] = {0};//父进程while (1){/* code */int len = read(pipefds[0],buf,sizeof(buf)); // 读端printf("parent receive : %s ,pid:%d\n",buf,getpid());}}else if(pid == 0){/* 子进程*/printf("i am child process,pid: %d\n",getpid());while (1){//向管道中写入数据char * str = "i am child";write(pipefds[1],str,strlen(str)); //写端sleep(1);}}return 0;
}

父子进程通信:

  

关系通信:互相发送

/*  
man pipe
#include <unistd.h>
int pipe(int pipefds[2]);创建一个匿名管道,用来进程间的通信
*/
#include <unistd.h>
#include <sys/types.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main(){//创建管道,在fork()之前int pipefds[2];     //得到两个文件描述符fd[0]和fd[1],分别指向管道的读端和写端。int ret = pipe(pipefds);if (ret == -1){perror("pipe");exit(0);}// 创建子进程pid_t pid = fork();if (pid > 0){printf("i am parent process,pid: %d\n",getpid());char buf[1024] = {0};//父进程while (1){/* code */int len = read(pipefds[0],buf,sizeof(buf)); // 读端printf("parent receive : %s ,pid:%d\n",buf,getpid());sleep(1);//向管道中写入数据char * str = "i am child";write(pipefds[1],str,strlen(str)); //写端sleep(1);}}else if(pid == 0){/* 子进程*/printf("i am child process,pid: %d\n",getpid());char buf[1024] = {0};while (1){//向管道中写入数据char * str = "i am child";write(pipefds[1],str,strlen(str)); //写端sleep(1);int len = read(pipefds[0],buf,sizeof(buf)); // 读端printf("child receive : %s ,pid:%d\n",buf,getpid());}}return 0;
}

 查看缓冲大小:ulimit -a

 pipe size:8块,每块512字节·,也就是4kb.

http://www.lryc.cn/news/11757.html

相关文章:

  • Reverse入门[不断记录]
  • 如何实现外网访问内网ip?公网端口映射或内网映射来解决
  • [acwing周赛复盘] 第 91 场周赛20230218
  • 蓝桥12届
  • 华为OD机试 - 斗地主(JS)
  • 【MyBatis】| MyBatis的注解式开发
  • python自制PDF转换.PNG格式图片(按每页生成图片完整源码)小工具!
  • Go 数组和切片反思
  • win10电脑性能优化设置
  • 作为初学者必须要了解的几种常用数据库!
  • 小红书日常实习一面面经
  • 将Nginx 核心知识点扒了个底朝天(一)
  • SSM项目搭建保姆级教程
  • LeetCode 350. 两个数组的交集 II
  • Python可以解码吗,解码打码是如何实现的
  • Jackson 序列化:Cannot deserialize value of type `java.time.LocalDateTime`
  • 机试_3_数据结构(一)_习题
  • 《Hadoop篇》------HDFS与MapReduce
  • 网络爬虫简介
  • 通过4个月的自动化学习,现在我也拿到了25K的offer
  • 分库分表了解
  • docker中 gitlab 安装、配置和初始化
  • 有哪些好用的C++Json库?
  • Docker 快速上手学习入门教程
  • 深度学习笔记:误差反向传播(1)
  • 锁相环(1)
  • 2023金三银四跳槽必会Java核心知识点笔记整理
  • 二十四节气—雨水,好雨知时节,当春乃发生。
  • 为什么要使用数据库?
  • 【原创】java+swing+mysql图书管理系统设计与实现