【0805作业】Linux中 AB终端通过两根有名管道进行通信聊天(半双工)(全双工)

news/2024/11/8 17:06:56/

作业一:打开两个终端,要求实现AB进程对话【两根管道】

打开两个终端,要求实现AB进程对话

  1. A进程先发送一句话给B进程,B进程接收后打印
  2. B进程再回复一句话给A进程,A进程接收后打印
  3. 重复1.2步骤,当收到quit后,要结束AB进程
  • 提示:两根管道

A终端代码

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>/***************************B终端***************************/  int main(int argc, const char *argv[])
{//创建管道1   A终端写入--->管道1--->B终端读取if(mkfifo("./fifo",0664) < 0){if(errno != 17){perror("mkfifo");return -1;}}printf("mkfifo pipe1 success __%d__\n",__LINE__);//创建管道2   B终端写入--->管道2--->A终端读取if(mkfifo("./myfifo",0664) < 0){if(errno != 17){perror("mkfifo");return -1;}}printf("mkfifo pipe2 success __%d__\n",__LINE__);//以读的方式打开管道1int fd_r=open("./fifo",O_RDONLY);if(fd_r < 0){perror("open");return -1;}printf("open pipe1 success __%d__\n",__LINE__);//以写的方式打开管道2int fd_w=open("./myfifo",O_WRONLY);if(fd_w < 0){perror("open");return -1;}printf("open pipe2 success __%d__\n",__LINE__);char buf[128]="";ssize_t res = 0;int c=-1;while(1){//管道1操作(读取数据)bzero(buf,sizeof(buf));res=read(fd_r,buf,sizeof(buf));c=strcmp(buf,"quit");//判断B终端输入的是否是quitif(0 == c){break;//是quit则退出进程}if(res < 0){perror("read");return -1;}if(0 == res ){printf("对方进程退出\n");break;}printf("A:%s\n",buf);//管道2操作(写入数据)printf("请输入>>> ");fgets(buf,sizeof(buf),stdin);buf[strlen(buf)-1] = '\0';if((write(fd_w,buf,sizeof(buf))) < 0){perror("write");return -1;}//当管道2关闭,管道2的写段尝试写入数据,则管道破裂,退出进程}close(fd_r);close(fd_w);return 0;
}

B终端代码

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>/***************************A终端***************************/int main(int argc, const char *argv[])
{//创建管道1    A终端写入--->管道1--->B终端读取if(mkfifo("./fifo",0664) < 0){if(errno != 17)//如果错误是已有管道,则跳过,可正常运行{perror("mkfifo");return -1;}}printf("mkfifo pipe1 success __%d__\n",__LINE__);//创建管道2    B终端写入--->管道2--->A终端读取if(mkfifo("./myfifo",0664) < 0){if(errno != 17){perror("mkfifo");return -1;}}printf("mkfifo pipe2 success __%d__\n",__LINE__);//以写的方式打开管道1int fd_w=open("./fifo",O_WRONLY);if(fd_w < 0){perror("open");return -1;}printf("open pipeA success __%d__\n",__LINE__);//以读的方式打开管道Bint fd_r=open("./myfifo",O_RDONLY);if(fd_r < 0){perror("open");return -1;}printf("open pipeB success __%d__\n",__LINE__);char buf[128]="";ssize_t res = 0;int c=-1;while(1){//管道1操作(写入数据)printf("请输入要对B说的话>>> ");fgets(buf,sizeof(buf),stdin);    //从终端获取数据buf[strlen(buf)-1] = '\0';   //将\n改成\0if((write(fd_w,buf,sizeof(buf))) < 0)  //将字符串写进管道A  {perror("write");return -1;}//当管道1的读段关闭,管道1的写段尝试写入数据,则管道破裂,退出进程//管道2操作(读取数据)bzero(buf,sizeof(buf));//清空字符串res=read(fd_r,buf,sizeof(buf));//读取B管道中的数据c=strcmp(buf,"quit");//将读到的数据与quit比较if(0 == c)//如果相同,c为0,达到退出条件,可以退出循环{break;}//printf("写入数据成功 res=%ld\n",res);if(res < 0)//read函数执行失败,返回负数{perror("read");return -1;}   if(0 == res)//read执行成功,但读到了0个数据`{printf("对方进程退出\n");break;}//打印从管道2中读取到的数据printf("B:%s\n",buf);}//关闭管道1、管道2close(fd_r);close(fd_w);return 0;
}

A终端结果

ubuntu@ubuntu:02_fifo$ gcc 03_pipe_w.c -o w
ubuntu@ubuntu:02_fifo$ ./w
mkfifo pipe1 success __22__
mkfifo pipe2 success __33__
open pipeA success __42__
open pipeB success __50__
请输入要对B说的话>>> 你好,我是A
B:你好呀,我是B
请输入要对B说的话>>> 你吃饭了吗?
B:吃了,你呢
请输入要对B说的话>>> 不告诉你
ubuntu@ubuntu:02_fifo$ 

B终端结果

ubuntu@ubuntu:02_fifo$ gcc 02_pipe_r.c -o r
ubuntu@ubuntu:02_fifo$ ./r
mkfifo pipe1 success __22__
mkfifo pipe2 success __33__
open pipe1 success __42__
open pipe2 success __50__
A:你好,我是A
请输入>>> 你好呀,我是B
A:你吃饭了吗?
请输入>>> 吃了,你呢
A:不告诉你
请输入>>> quit
对方进程退出
ubuntu@ubuntu:02_fifo$ 

fd741777d8ed4957bfa812ba1a3af75e.png

作业二:在作业一的基础上,AB能随时发信息给对方(全双工)

此代码无法quit结束对话

A终端代码

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
#include <wait.h>/***************************A终端***************************/int main(int argc, const char *argv[])
{umask(0);//创建管道1    A终端写入--->管道1--->B终端读取if(mkfifo("./fifo",0777) < 0){if(errno != 17)//如果错误是已有管道,则跳过,可正常运行{                                                                                      perror("mkfifo");return -1;}}printf("mkfifo pipe1 success __%d__\n",__LINE__);umask(0);//创建管道2    B终端写入--->管道2--->A终端读取if(mkfifo("./myfifo",0664) < 0){if(errno != 17){perror("mkfifo");return -1;}}printf("mkfifo pipe2 success __%d__\n",__LINE__);//创建子进程pid_t cpid=fork();//以写的方式打开管道1int fd_w=open("./fifo",O_WRONLY);if(fd_w < 0){perror("open");return -1;}printf("open pipeA success __%d__\n",__LINE__);//以读的方式打开管道Bint fd_r=open("./myfifo",O_RDONLY);if(fd_r < 0){perror("open");return -1;}printf("open pipeB success __%d__\n",__LINE__);char buf[128]="";ssize_t res = 0;int c=-1;if(0 == cpid)//子进程发送{while(1){//管道1操作(写入数据)fgets(buf,sizeof(buf),stdin);    //从终端获取数据buf[strlen(buf)-1] = '\0';   //将\n改成\0if((write(fd_w,buf,sizeof(buf))) < 0)  //将字符串写进管道A  {perror("write");return -1;}printf("A: %s\n",buf);//当管道1的读段关闭,管道1的写段尝试写入数据,则管道破裂,退出进程//c=strcmp(buf,"quit\n");//将读到的数据与quit比较if(strncmp(buf,"quit",4) == 0)//如果相同,c为0,达到退出条件,可以退出循环{break;}//printf("写入数据成功 res=%ld\n",res);}close(fd_w);exit(0);}else if(cpid > 0)//父进程接收{while(1){//管道2操作(读取数据)bzero(buf,sizeof(buf));//清空字符串res=read(fd_r,buf,sizeof(buf));//读取B管道中的数据if(res < 0)//read函数执行失败,返回负数{perror("read");return -1;}if(0 == res)//read执行成功,但读到了0个数据`{printf("对方进程退出\n");break;}//打印从管道2中读取到的数据printf("B:%s\n",buf);}wait(NULL);close(fd_r);}//关闭管道1、管道2close(fd_r);close(fd_w);return 0;
}

B终端代码

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
#include <wait.h>/***************************B终端***************************/int main(int argc, const char *argv[])
{//创建管道1   A终端写入--->管道1--->B终端读取if(mkfifo("./fifo",0777) < 0){if(errno != 17){perror("mkfifo");return -1;}}printf("mkfifo pipe1 success __%d__\n",__LINE__);//创建管道2   B终端写入--->管道2--->A终端读取if(mkfifo("./myfifo",0777) < 0){if(errno != 17){perror("mkfifo");return -1;}}printf("mkfifo pipe2 success __%d__\n",__LINE__);//创建子进程pid_t cpid=fork();//以读的方式打开管道1int fd_r=open("./fifo",O_RDONLY);if(fd_r < 0){perror("open");                                                                  return -1;}printf("open pipe1 success __%d__\n",__LINE__);//以写的方式打开管道2int fd_w=open("./myfifo",O_WRONLY);if(fd_w < 0){perror("open");return -1;}printf("open pipe2 success __%d__\n",__LINE__);char buf[128]="";ssize_t res = 0;//int c=-1;if(0 == cpid)//子进程接收{while(1){//管道1操作(读取数据)bzero(buf,sizeof(buf));res=read(fd_r,buf,sizeof(buf));if(res < 0){perror("read");return -1;}if(0 == res ){printf("对方进程退出\n");break;}printf("A:%s\n",buf);}close(fd_r);exit(0);}else if(cpid > 0)//父进程发送{while(1){//管道2操作(写入数据)printf("B: ");fgets(buf,sizeof(buf),stdin);buf[strlen(buf)-1] = '\0';if((write(fd_w,buf,sizeof(buf))) < 0){perror("write");return -1;}//当管道2关闭,管道2的写段尝试写入数据,则管道破裂,退出进程//c=strcmp(buf,"quit");if(strncmp(buf,"quit",4) == 0)//判断B终端输入的是否是quit{break;}}close(fd_w);wait(NULL);}else{perror("fork");return -1;}close(fd_r);close(fd_w);return 0;
}


http://www.ppmy.cn/news/1007893.html

相关文章

跨域+四种解决方法

文章目录 一、跨域二、JSONP实现跨域请求三、前端代理实现跨域请求四、后端设置请求头实现跨域请求五、Nginx代理实现跨域请求5.1 安装Nginx软件5.2 使用Ubuntu安装nginx 本文是在学习课程满神yyds后记录的笔记&#xff0c;强烈推荐读者去看此课程。 一、跨域 出于浏览器的同…

常用消息中间件对比

Kafka 1.基于Pull的模式来处理消息消费 2.追求高吞吐量 3.一开始的目的就是日志收集和传输 4.0.8版本开始支持复制&#xff0c;不支持事务&#xff0c;对消息的重复、丢失、错误没有严格要求、适合产生大量数据的互联网服务的数据收集业务. RabbitMQ RabbitMQ是使用Erlang语…

Linux软件实操

systemctl命令 Linux系统的很多内置或第三方的软件均支持使用systemctl命令控制软件(服务)的启动、停止、开机自启 systemctl start(启动) 或 stop(关闭) 或 status(查看状态) 或 enable(开启开机自启) disable(关闭开机自启) 服务名: 控制服务的状态 系统内置的服务: Netwo…

第一课-前提-Stable Diffusion 教程

学习 SD 的前提是电脑配置! SD 参考配置: 建议选择台式机 i5 CPU, 内存16GB,N卡 RTX3060, 8G显存以上的配置(最低配) 在此基础上的配置越高越好。 比如,cpu i7 更好,显卡能有 RTX4090 更好,32显存要能有最好,嘿嘿嘿。 如何查看自己的显卡配置? Win+R 输入 “dxdiag…

【力扣】21. 合并两个有序链表 <链表指针>

【力扣】21. 合并两个有序链表 将两个升序链表合并为一个新的升序链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 示例1 输入&#xff1a;l1 [1,2,4], l2 [1,3,4] 输出&#xff1a;[1,1,2,3,4,4] 示例 2 输入&#xff1a;l1 [], l2 [] 输出&#xff1a;…

关于java异常的整理

文章目录 一、异常分类二、throw、throws、try-catch-finally三、CglibAopProxy中对异常的处理4、关于UndeclaredThrowableException 一、异常分类 java异常层级结构 Throwable:所有异常的根接口 Error:严重错误,程序无法处理和恢复 例如VirtualMachineError,OOMError等 Excep…

ABP 后台任务生产者-消费者模式的实现逻辑

ABP 的后台任务是通过生产者-消费者模式来实现的。 在 ABP 中&#xff0c;生产者通过调用 IBackgroundJobManager 接口的 AddJobAsync 方法来添加后台任务&#xff0c;消费者则通过调用 IBackgroundJobExecuter 接口的 ExecuteAsync 方法来执行后台任务。当生产者添加一个后台…

Java阶段五Day21

Java阶段五Day21 文章目录 Java阶段五Day21问题解析rocketmq清空数据 linux学习背景什么是linux系统虚拟机介绍启动 虚拟机linux虚拟机网络的问题 linux系统的基础命令命令提示符命令格式pwd指令ls指令cd指令mkdirtouch指令cp指令rm指令mv指令cat指令tail指令 文本编辑器vim操作…