使用fifo实现一个服务器端多个客户端通信

有名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写端(这里还不是很清楚)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章