目录
有名管道实现简单版聊天功能
有名管道相关知识
- 创建管道:
mkfifo 名字
- 一旦使用mkfifo创建了一个管道,我们就可以用open打开,常见的I/O函数都可用于fifo,例如,close, read, write, unlink
- FIFO 严格遵守先进先出的规则,对管道及FIFO 的读总是从开始处返回数据,对他们的写则是添加到末尾。但是不支持lseek()等文件定位操作。
实现流程
进程A:
- 以只写的方式打开管道1;
- 以只读的方式打开管道2;
- 循环的写读数据。
进程B:
- 以只读的方式打开管道1;
- 以只写的方式打开管道2;
- 循环的读写数据。
具体的代码如下:
实现代码
//chatA.c
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
int main() {
// 1.判断有名管道文件是否存在
int ret = access("fifo1", F_OK);
if(ret == -1) {
// 文件不存在
printf("管道不存在,创建对应的有名管道\n");
ret = mkfifo("fifo1", 0664);
if(ret == -1) {
perror("mkfifo");
exit(0);
}
}
ret = access("fifo2", F_OK);
if(ret == -1) {
// 文件不存在
printf("管道不存在,创建对应的有名管道\n");
ret = mkfifo("fifo2", 0664);
if(ret == -1) {
perror("mkfifo");
exit(0);
}
}
// 2.以只写的方式打开管道fifo1
int fdw = open("fifo1", O_WRONLY);
if(fdw == -1) {
perror("open");
exit(0);
}
printf("打开管道fifo1成功,等待写入...\n");
// 3.以只读的方式打开管道fifo2
int fdr = open("fifo2", O_RDONLY);
if(fdr == -1) {
perror("open");
exit(0);
}
printf("打开管道fifo2成功,等待读取...\n");
char buf[128];
// 4.循环的写读数据
while(1) {
memset(buf, 0, 128);
// 获取标准输入的数据
fgets(buf, 128, stdin);
// 写数据
ret = write(fdw, buf, strlen(buf));
if(ret == -1) {
perror("write");
exit(0);
}
// 5.读管道数据
memset(buf, 0, 128);
ret = read(fdr, buf, 128);
if(ret <= 0) {
perror("read");
break;
}
printf("buf: %s\n", buf);
}
// 6.关闭文件描述符
close(fdr);
close(fdw);
return 0;
}
//chatB.c
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<stdlib.h>
#include<fcntl.h>
#include<string.h>
int main(){
//1、判断有名管道文件fifo1是否存在
int ret = access("fifo1",F_OK);
if(ret == -1){
//管道文件不存在
printf("管道不存在,创建对应有名管道文件\n");
int ret = mkfifo("fifo1",0664);
if(ret = -1){
perror("mkfifo");
exit(0);
}
}
//判断有名管道文件fifo2是否存在
ret = access("fifo2",F_OK);
if(ret == -1){
//管道文件不存在
printf("管道不存在,创建对应有名管道文件\n");
int ret = mkfifo("fifo2",0664);
if(ret = -1){
perror("mkfifo");
exit(0);
}
}
//2.以只读的方式打开管道fifo1
int fdr = open("fifo1",O_RDONLY);
if(fdr == -1){
perror("open");
exit(0);
}
printf("打开fifo1成功,等待读入数据\n");
//3.以只写的方式打开管道fifo2
int fdw = open("fifo2",O_WRONLY);
if(fdw == -1){
perror("open");
exit(0);
}
printf("打开fifo2成功,等待写入数据\n");
char buf[128];
//4.循环读数据
while(1){
//清空数据
memset(buf,0,128);
//读数据
ret = read(fdr,buf,128);
if(ret <= 0){
perror("read");
break;
}
printf("buf:%s\n",buf);
//5.循环写读数据
//清空数据
memset(buf,0,128);
//获取标准输入数据
fgets(buf,128,stdin);
//写入数据
ret = write(fdw,buf,strlen(buf));
if(ret == -1){
perror("write");
exit(0);
}
}
//关闭读写管道
close(fdr);
close(fdw);
return 0;
}