TCP/IP网络编程 第十一章:进程间通信

进程间通信的基本概念

在上一章中我们讲到,进程自身有独立的内存空间,进程之间是相互独立的存在。因此在没有任何机制的支持下,我们可以将进程看作相互孤立的存在。

但是进程之间在某种程度上也是要“交流”的,下面正式介绍进程间通信的方法.

通过管道实现进程间通信

为了完成进程间通信,需要创建管道。管道并非属于进程的资源,而是和套接字一样,属于操作系统(也就不是fork函数的复制对象)。所以,两个进程通过操作系统提供的内存空间进行通信。下面介绍创建管道的函数。

#include<unistd.h>
int pipe(int filedes[2]);//成功时返回0,失败时返回-1。filedes[0]  //通过管道接收数据时使用的文件描述符,即管道出口。filedes[1]  //通过管道传输数据时使用的文件描述符,即管道入口。

以2个元素的int数组地址值作为参数调用上述函数时,数组中存有两个文件描述符,它们将被用作管道的出口和入口。父进程调用该函数时将创建管道,同时获取对应于出入口的文件描述符,此时父进程可以读写同一管道(相信大家也做过这样的实验)。但父进程的目的是与子选程进行数据交换,因此需要将入口或出口中的1个文件描述符传递给子进程。如何完成传递呢?答案就是调用fork函数。通过下列示例进行演示。

#include <stdio.h>
#include <unistd.h>#
define BUF_SIZE 30int main(int argc, char *argv[]){int fds[2];char str[]="who are you?";char buf[BUF_SIZE];pid_t pid;pipe(fds);pid=fork();if(pid==0)write(fds[1], str, sizeof(str));else{read(fds[0], buf, BUF_SIZE);puts(buf);}return 0;
}

上述代码的重点在于,父子进程都可以访问管道的IO路径,但子进程仅用输入路径,父进程仅用输出路径。

通过管道进行进程间双向通信

先给出示例代码稍作讨论

#include<stdio.h>
#include<unistd.h>
#define BUF_SIZE 30int main(int argc, char *argv[]){int fds[2];char str1[]="who are you?";char str2[]="Thank you for your message";char buf[BUF_SIZE];pid_t pid;pipe(fds);pid=fork();if(pid==0){write(fds[1], str1, sizeof(str1));sleep(2);read(fds[0], buf, BUF_SIZE);printf("Child proc output: %s \n",buf);}else{read(fds[0], buf, BUF_SIZE);printf("Parent proc output: %s \n",buf);write(fds[1], str2, sizeof(str2));sleep(3);}
return 0;
}

运行结果应该和大家的预想一致。这次注释第18行代码后再运行(务必亲自动手操作)。虽
然这行代码只将运行时间延迟了2秒,但已引发运行错误。产生原因是什么呢?
“向管道传递数据时,先读的进程会把数据取走。”

这里管程是操作系统分配的一个地址空间,它不属于那两个进程。那么两个程序在运行时,一个进程将内容放入这个"无主之地"。如果放入的这一方不等待几秒而是直接往下运行,那么反而它自己放入的内容被自己取走了。时间片耗尽后,轮到另一个进程时,它便会被无限期的堵塞在读取的函数中,因此产生错误。

从上述示例中可以看到,只用1个管道进行双向通信并非易事。为了实现这一点,程序需要预测并控制运行流程,这在每种系统中都不同,可以视为不可能完成的任务。既然如此,该如何进行双向通信呢?
“创建2个管道。”

非常简单,1个管道无法完成双向通信任务,因此需要创建2个管道,各自负责不同的数据流
动即可。

下面是示例代码

#include<stdio.h>
#include<unistd.h>
#define BUF_SIZE 30int main(int argc, char *argv[]){int fds1[2], fds2[2];char stri[]="who are you?";char str2[]="Thank you for your message";char buf[BUF_SIZE];pidt pid;pipe(fds1), pipe(fds2);pid=fork();if(pid==0){write(fds1[1], str1, sizeof(str1));read(fds2[0], buf, BUF_SIZE);printf("Child proc output: %s \n", buf);}else{read(fds1[0], buf, BUF_SIZE);printf("Parent proc output: %s \n", buf);write(fds2[1], str2, sizeof(str2));sleep(3);}
return 0;
}

运用进程间通信

保存消息的回声服务器端

下面扩展第10章的回声服务器端,添加如下功能:“将回声客户端传输的字符串按序保存到文件中。”


我希望将该任务委托给另外的进程。换言之,另行创建进程,从向客户端提供服务的进程读
取字符串信息。当然,该过程中需要创建用于接收数据的管道。
下面给出示例。该示例可以与任意回声客户端配合运行。

#include <'头声明与第10章的示例一致。'>
#define BUF_SIZE 100
void error_handling(char *message);
void read_childproc(int sig);int main(int argc, char *argv[]){int serv_sock, clnt_sock;struct sockaddr_in serv_adr, cint_adr;int fds[2];pid_t pid;struct sigaction act;socklen_t adr_sz;int str_len, state;char buf[BUF_SIZE];if(argc!=2) {printf("Usage : %s <port>\n", argv[0]);exit(1);}act.sa_handler=read_childproc;sigemptyset(&act.sa_mask);act.sa_flags=0;state=sigaction(SIGCHLD, &act, 0);serv_sock=socket(PF_INET, SOCK_STREAM, 0);memset(&serv_adr, 0, sizeof(serv_adr));serv_adr.sin_family=AF_INET;serv_adr.sin_addr.s_addr=htonl(INADDR_ANY);serv_adr.sin_port=htons(atoi(argv[1]));if(bind(serv_sock,(struct sockaddr*)&serv_adr, sizeof(serv_adr))==-1)error_handling("bind() error");if(listen(serv_sock,5)==-1)error_handiing("listen() error");pipe(fds);pid=fork();if(pid==0){FILE *fp=fopen("echomsg.txt","wt");char msgbuf[BUF_SIZE];int i, len;for(i=0;i<10;i++){len=read(fds[0], msgbuf, BUF_SIZE);fwrite((void*)msgbuf,1, len, fp);}fclose(fp);return 0;}while(1){adr_sz=sizeof(clnt_adr);clnt_sock=accept(serv_sock,(struct sockaddr*)&clnt_adr, &adr_sz);if(clnt_sock==-1)continue;elseputs("new client connected...");pid=fork();if(pid==0){close(serv_sock);while((str_len=read(clnt_sock, buf, BUF_SIZE))!=0){write(clnt_sock, buf, str_len);//回声write(fds[1], buf, str_len);   //记录消息}close(clnt_sock);puts("client disconnected...");return 0;}else close(clnt_sock);}close(serv_sock);return 0;
}void read_childproc(int sig){
//与上一章示例一致,故省略。
}void error_handling(char* message){
//与上一章示例一致,故省略。
}

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

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

相关文章

在uni中使用vue3写h5的pdf导出

先安装依赖 npm install --save html2canvas npm install jspdf --save 把dom转canvas&#xff0c;屏幕截图基于 DOM&#xff0c;因此可能不是 100% 准确到真实表示&#xff0c;因为它不会制作实际的屏幕截图&#xff0c;而是根据页面上可用的信息构建屏幕截图。 components…

linux 一键安装docker docker-compose

# 环境 deploy.sh docker-19.03.9.tgz docker-compose-linux-x86_64 tar -zxvf docker-19.03.9.tgz mv docker-19.03.9 docker cp docker/* /usr/bin/ touch /etc/systemd/system/docker.service cat << EOG > /etc/systemd/system/docker.service [Unit] Descripti…

【技巧】Maven重复依赖分析查找

【技巧】Maven重复依赖分析查找 遇到奇葩的错误可以考虑是不是依赖冲突了 比如同一段代码 再这个项目中好好的 另一个项目中不能用等 idea安装插件 maven helper 打开pom文件 输入要查找的依赖 将不用的排除掉 右键排除即可

“深入解析API接口:原理、用途和最佳实践“

API&#xff08;Application Programming Interface&#xff0c;应用程序编程接口&#xff09;是不同软件或系统之间进行交互的桥梁&#xff0c;它提供了一组定义好的规则和功能&#xff0c;使得不同的应用程序能够相互通信和交换数据。API接口的理解、应用和最佳实践对于开发者…

在阿里云平台注册一个域名

我们访问阿里云官网 阿里云 然后 我们右上角点击登录 然后 按正常操作流程登录 登录成功后 我们点击控制台 我们将鼠标 移入 右上角 图片指向的位置 我们点击域名 进入界面后点击注册域名 在输入框中输入域名内容 然后 按回车 然后弹出的列表 我们可以选一个未注册的 点击…

解决Linux打开文件数过多error:too many open files

解决Linux打开文件数过多error:too many open files #查看当前的最大打开文件数 ulimit -n #默认只有1024,当负载较大的服务器时&#xff0c;很容易遇到error: too many open files。因此&#xff0c;需要将其改大。#可临时修改&#xff0c;但重启后就无效 ulimit -n 65535 #查…

Druid-排查conditionDoubleConstAllow配置问题(double const condition)

Druid-排查conditionDoubleConstAllow配置问题(double const condition) 报错信息 Caused by: java.sql.SQLException: sql injection violation, dbType postgresql, druid-version 1.2.18, double const condition : SELECT * FROM test where 11 AND TRUE AND TRUE关键词&…

正则表达式概念以及语法的使用

目录 1.概念 2. 为什么使用正则表达式&#xff1f; 3. 语法 1.普通字符 非打印字符 2. 特殊字符 3. 限定符 4. 定位符 5. 运算优先级 3.匹配规则 1. 基本模式匹配 2. 字符簇 3. 确定重复出现 1.概念 正则表达式(Regular Expression)是一种文本模式&#xff0c;包…

【Docker】docker-compose基本使用

【Docker】docker-compose基本使用 文章目录 【Docker】docker-compose基本使用1. docker 项目部署弊端2. docker-compose 简介3. 实践4. 模板命令4.1 build 指令4.2 command 指令4.3 container_name 指令4.4 depends_on 指令4.5 env_file 指令4.6 environment 指令4.7 image 指…

基于Redisson的Redis分布式锁

Redisson分布式锁_redissonclient_甩锅虾的博客-CSDN博客 *** 分布式锁*/ public interface DistributedLock {/*** 尝试获取锁* param lockName* param seconds* return*/Boolean tryAcquire(String lockName, long seconds, TimeUnit unit);/*** 释放锁* param lockKey*/v…

如何使用自动化构造随机路由模型

为什么要仿真随机路由&#xff1f; 路由器测试中&#xff0c;为了最大程度还原现网路由情况&#xff0c;评估路由器在现网环境下稳定工作各项指标&#xff0c;需要对导入路由进行离散仿真&#xff0c;目前路由仿真可分为导入路由与生成路由两种方式&#xff0c;导入路由需要现…

vue如何使用webscorket实现多人协同在线表格填写

要使用WebSocket在Vue中实现多人协同在线表格填写&#xff0c;你可以按照以下步骤进行操作&#xff1a;1. 安装WebSocket库&#xff1a;首先&#xff0c;在Vue项目中安装WebSocket库。你可以使用 socket.io-client 库来处理WebSocket通信。使用npm或yarn安装它&#xff1a;npm …

学习遇上的一点补救

从开始学习java到现在逐步确定好发展路线。是时候停一下步伐了。 之前不停的往前走&#xff0c;从学校的课程、书本开始&#xff0c;没有借助网课、视频&#xff0c;依靠自己一步一个脚印。从Java实用程序、Javaweb、JavaEE然后选择MySQL数据库、然后觉得有必要对前端了解&…

think-on-graph: 基于知识图谱的大模型推理

概述 本文的研究背景是大规模语言模型在复杂推理任务中存在困难并展示了较低的性能&#xff0c;特别是在需要知识的追溯能力、及时性和准确性的场景中。 过去的方法主要面临两个问题&#xff1a;推理不负责任容易生成虚构或带有有害文本&#xff0c;以及模型在预训练阶段无法…

【深入浅出C#】章节 5: 高级面向对象编程:泛型编程和集合类型

高级面向对象编程是在基础面向对象编程的基础上进一步深入和拓展的一种编程范式。它强调封装、继承和多态的概念&#xff0c;并引入了泛型编程和集合类型等高级特性。高级面向对象编程提供了更灵活、可扩展和可复用的代码结构&#xff0c;能够帮助开发者构建更复杂、更高效的应…

基于IPC-CFX的点对点通信C#

IPC-CFX有两种主要的通信方式&#xff0c;可以通过RabbitMQ发布和订阅&#xff0c;也可以通过request和response进行点对点的通信&#xff0c;本文主要讲的是点对点的通信方式。 在vscode里建立新的dotnet项目&#xff0c;可以通过终端输入dotnet new console来建立&#xff0c…

【拿来就能用】C#指定打印机打印的类

之前写过一个“C#WinForm程序中选择打印机打印”的文章&#xff0c;但在使用过程中&#xff0c;尤其是生成Word文档时&#xff0c;会感觉系统响应较慢。如果不需要留存打印文档的电子版&#xff0c;可以使用下面的类直接打印。相比之前的方法&#xff0c;这种方法更简单&#x…

MySQL常用语句大全

语句 DDL MySQL的DDL&#xff08;数据定义语言&#xff09;是一组用于创建、修改和删除数据库、表、索引、视图、存储过程和触发器等数据库对象的语句。下面是一些常用的MySQL DDL语句和它们的详细说明&#xff1a; alter 在MySQL中&#xff0c;DDL&#xff08;数据定义语言…

LCD—STM32液晶显示(2.使用FSMC模拟8080时序)

目录 使用STM32的FSMC模拟8080接口时序 FSMC简介 FSMC NOR/PSRAM中的模式B时序图 用FSMC模拟8080时序 重点&#xff1a;HADDR内部地址与FSMC地址信号线的转换&#xff08;实现地址对齐&#xff09; 使用STM32的FSMC模拟8080接口时序 ILI9341的8080通讯接口时序可以由STM32使…

北邮国院物联网 Microprocessor 微处理器笔记

Introduction-随便聊 嵌入式系统是什么&#xff1f;专用的计算机系统。为专门功能可能对计算机架构&#xff0c;外设等做出一些取舍。 通常的限制&#xff1a;Cost&#xff08;比如大量部署传感器节点&#xff09;&#xff0c;Size and weight limits&#xff08;特定应用场景…