I/O多路复用select服务器

一、五种I/O模型
linux下的I/O模型分为五种模型:分别是阻塞式I/O、非阻塞式I/O、I/O多路复用(多路转接)、信号驱动I/O(SIGIO)、异步I/O。
这五种I/O由分为两类:一类是同步I/O包括了前面四种,还有一类是异步I/O。
我们把I/O是分成两步的,第一步是“等”,第二步就是数据的搬迁。而对于同步I/O和异步I/O来说它们之间的区别就是同步I/O的数据搬迁工作要自己来完成,而异步I/O的数据搬迁工作是别人来完成的。而阻塞I/O和非阻塞I/O的最大不同就是“等”的方式是不同的。

二、select函数介绍
select系统调用的用途是:在一段指定的时间内,监听用户感兴趣的文件描述符上的可读、可写、异常事件。
我们看一下select函数的原型:
这里写图片描述
我们可以看到select函数里面有五个参数,下面一次做一个说明:
1、nfds:指定被监听的文件描述符的总数,通常设置为select监听的所有文件描述符的中的最大值再加1(文件描述符是从0开始的)。
2、readfds、writefds、exceptfds这三个参数都是输入输出型参数,它们表示的是可读、可写、异常事件的文件描述符的集合。这三个参数做输入型参数的意思是:关心特定的文件描述符上的指定事件;这三个参数作为输出型参数的意思是:关心的事件发生了变化已经就绪。这三个参数的类型都是fd_set的指针类型的结构体,fd_set结构体仅包含一个整形数组,数组的每一位元素的每一位(bit)标记了一个文件描述符。fd_set所能包含的文件描述符的数量由FD_SETSIZE指定,这就限制了select处理的文件描述符的总量,这样select的服务就是有上限的。
我们在访问fd_set这个数据结构的时候还有几个宏来访问fd_set,集体如下:
FD_ZERO(fd_set *fdset) //清除fdset所有位
FD_SET(int fd, fd_set *fdset) //设置fdset位
FD_CLR(int fd, fd_set *fdset) //清除fdset位
int FD_ISSET(nt fd, fd_set *fdset)//检查fdset是否被设置
3、timeout:
这里写图片描述
用来设置select函数的超时时间,select给我们提供了而一个微秒级的计时方式,如果timeout的成员tv_sec和tv_usec的成员都为0,select就会立即返回。
4、返回值:
select成功时就会返回就绪(可读、可写、异常)的文件描述符的总数,在超时时间内没有任何文件描述符就绪返回0,如果select失败就会返回-1,设置errno为EINTR。

三、代码展示
select_server.c:

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/select.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<fcntl.h>
#include<unistd.h>

int fds[sizeof(fd_set)*8];

static usage(const char* proc)
{
    printf("usage :%s [local_ip] [local_port]\n",proc);
}

int startup(const char* ip,int port)
{
    int sock = socket(AF_INET,SOCK_STREAM,0);
    if(sock < 0)
    {
        perror("socket");
        exit(2);
    }
    int opt = 1;
    setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));

    struct sockaddr_in local;
    local.sin_family = AF_INET;
    local.sin_port = htons(port);
    local.sin_addr.s_addr = inet_addr(ip);//点分十进制化为四字节

    if(bind(sock,(struct sockaddr*)&local,sizeof(local)) < 0)
    {
        perror("bind");
        exit(3);
    }

    if(listen(sock,10) < 0)
    {
        perror("listen");
        exit(4);
    }
    return sock;
}

int main(int argc,char* argv[])
{
    if(argc != 3)
    {
        usage(argv[0]);
        return 1;
    }
    int listen_sock = startup(argv[1],atoi(argv[2]));
    printf("fd_set: %d\n",sizeof(fd_set)*8);
    int fds[sizeof(fd_set)];
    int nums = sizeof(fds)/sizeof(fds[0]);
    int i = 0;
    for(; i < nums; i++)
    {
        fds[i] = -1;
    }
    fds[0] = listen_sock;
    int maxfd = -1;
    fd_set rfds;//读事件
    fd_set wfds;//写事件
    while(1)
    {
        int maxfd = -1;
        struct timeval timeout = {2,0};
        FD_ZERO(&rfds);
        FD_ZERO(&wfds);
        i = 0;
        for(; i < nums;i++)
        {
            if(fds[i] == -1)
            {
                continue;
            }
            FD_SET(fds[i],&rfds);
            if(maxfd < fds[i])
            {
                maxfd = fds[i];
            }
        }
        switch(select(maxfd+1,&rfds,&wfds,NULL,&timeout))
        {                      
            case -1:   //select失败
                perror("select");
                break;
            case 0:    //超过时间没有任何描述符就绪
                printf("time out!\n");
                break;
            default:   //成功
                {
                //at least one fd ready!
                i = 0;
                for(; i < nums;i++)
                {
                    if(i == 0 && FD_ISSET(fds[i],&rfds))//listen_sock is ready, get connect
                    {
                        struct sockaddr_in client;
                        socklen_t len = sizeof(client);
                        int new_sock = accept(listen_sock,(struct sockaddr*)&client,&len);
                        if(new_sock < 0)
                        {
                            perror("accept");
                            continue;
                        }
                        //accept只是获得了一个新的连接  但是不能保证读和写正常
                        //如果对面一直不发数据  那么就会一直处于read里的等待情况  就会导致服务器的不能正常运行

                        printf("get a new client: [%s:%d]\n",inet_ntoa(client.sin_addr),ntohs(client.sin_port));


                        int j = 1;
                        for(; j < nums ; j++)
                        {
                            if(fds[j] == -1)
                            {
                                break;
                            }
                        }
                        if(j == nums)//已经达到所能监听的文件描述符的最大值 只能关闭
                        {
                            close(new_sock);
                        }
                        else
                        {
                            fds[j] = new_sock;
                        }
                    }
                    else if(i != 0 && FD_ISSET(fds[i],&rfds))//normal fd is ready
                    {
                        char buf[1024];
                        ssize_t s = read(fds[i],buf,sizeof(buf)-1);
                        if( s > 0)
                        {
                            buf[s] = 0;
                            printf("client# %s\n",buf);
                            FD_SET(fds[i],&wfds);

                        }
                        else if(s == 0)
                        {
                            printf("client is quit!\n");
                            close(fds[i]);
                            fds[i] = -1;
                        }
                        else
                        {
                            perror("read");
                            close(fds[i]);
                            fds[i] = -1;
                        }
                    }
                    if(i !=0 && FD_ISSET(fds[i],&wfds))//普通的写操作
                    {
                        const char* msg = "hello client!\n";
                        ssize_t s = write(fds[i],msg,strlen(msg));
                        if(s < 0)
                        {
                            perror("write");
                        }
                        else
                        {
                            FD_CLR(fds[i],&wfds);
                        }
                    }
                }
                break;
            }
        }

    }
    close(listen_sock);
    return 0;
}

select_client.c:

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/select.h>
#include<arpa/inet.h>
#include<fcntl.h>
#include<unistd.h>
#include<netinet/in.h>

static usage(const char* proc)
{
    printf("usage: [client_ip] [client_port]%s\n",proc);
}

int main(int argc,const char* argv[])
{
    if(argc != 3)
    {
        usage(argv[0]);
        return 1;
    }

    int sock = socket(AF_INET,SOCK_STREAM,0);
    if(sock < 0)
    {
        perror("socket");
        exit(2);
    }

    struct sockaddr_in server;
    server.sin_family = AF_INET;
    server.sin_port =htons(atoi(argv[2]));
    server.sin_addr.s_addr = inet_addr(argv[1]);


    if(connect(sock,(struct sockaddr*)&server,sizeof(server)) < 0)
    {
        perror("connect");
        exit(3);
    }

    char  buf[1024];
    int fd;
    while(1)
    {
        printf("please Enter#: ");
        fflush(stdout);
        fd = dup(1);

        ssize_t s = read(0,buf,sizeof(buf)-1);
        if(s > 0)
        {
            buf[s-1] = 0;

            close(1);
            dup2(sock,1);
            printf("%s",buf);
            fflush(stdout);
        }
        else
        {
            perror("read");
            exit(4);
        }
        dup2(fd,1);
        ssize_t _s = read(sock,buf,sizeof(buf)-1);
        if(_s > 0)
        {
            buf[_s] = 0;
            printf("server echo#: %s\n",buf);
        }
    }
    close(fd);
    close(sock);
    return 0;
}

四、运行结果
服务器端运行情况:
这里写图片描述
客户端的运行情况:
这里写图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值