有名fifo默认是阻塞读,阻塞写的,如果要设置阻塞属性,可以再open()的时候进行设置
在阻塞情况下:实现服务器端可以不断读取客户端写到管道里面的文件流
这个里面有个重要的问题时,如果使用服务器端进行read(),读的buf的大小是多少呢?如果每一个客户端每次传过来的buf大小都不一样,那么服务器端使用read函数必然会造成数据读取不完整或者超出实际写进去的,这是因为,对于多个进程都在向fifo文件中写时,一个进程一个进程将要写的内容写进去,如果在读buf大小的数据的时候,有两个进程一前一后写入数据(在下次read()之前),那么这两个进程的写入数据就会混乱的读入读进程。
为了解决这一问题:
关键:统一buf的大小,也就是write read函数每次操作的buf的大小都是确定的,这样的话,每一个进程的数据就有了“原子性”,被封装在一个buf中,保证每一次读,都是一个完整的数据包
比如:
客户端有登录包,发送信息给其他客户端的信息包等等,将不同包的内容封装,这里就像一个协议,不同的数据包有不同的编号,比如,登录包,编号为1,信息包编号为2,这样,讲将要传递的内容拼接在一起,作为长度固定的整体,如果长度不够,填充特殊字符,就像ip数据报一样,这样,按照不同编号,服务器也对应了不同的解析buf的方法,这样就保证多个进程的数据可以完整的传递到服务器端了
代码:
client_0
#include <stdio.h> #include <unistd.h> #include <error.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <stdlib.h> #define BUF_SIZE 1024 void sys_err(char *errmes,int errno){ perror(errmes); exit(errno); } int main(int argc ,char **argv){ pid_t pid; int fd,i; char buf[BUF_SIZE]; if(argc<2){ printf("usage:./client filename"); exit(1); } if((fd=open(argv[1],O_WRONLY))<0){ sys_err("open",3); } while(1){ write(fd,"helloclient_0",13); //将buf的长度定为13个字节 sleep(1); } close(fd); return 0; } |
server.c
int main(int argc ,char **argv){ pid_t pid; int fd,n; int buf[BUF_SIZE]; if(argc<2){ printf("usage:./server filename"); exit(1); } //创建一个fifo //先判断该文件是否存在,如果不存在再创建,存在就不创建了 if(access(argv[1],F_OK)==-1){ if(mkfifo(argv[1],0777)<0){ sys_err("mkfifo",2); } } //打开该fifo if((fd=open(argv[1],O_RDONLY))<0){ sys_err("open",3); } //注册信号 signal(SIGCHLD,sig_child); //从fifo循环接收传递进来的数据 while(1){ if((n=read(fd,&buf,13))){ //read函数的buf也是13个字节 printf("%s\n",buf); }else{ continue; } sleep(1); } close(fd); return 0; } |
上面的两段代码,只是简单的表示了一下文字的代码形式,关于实际的协议定义,还需要推敲,并且buf的大小也要根据协议而定;对于不同的
协议,有不同的拼接填充方式以及解析方式,代码中只是简单实现了
1.使用mkfifo()函数新建一个fifo文件
2.使用open()打开函数打开该文件(可设置阻塞属性)
3.向文件中读写
ps:如果只有一个读端,并且先关闭读端的话,写端将进程也将结束
如果先没有fifo读端,无法运行fifo写端(这里还不是很清楚)