网络编程——I/O复用

参考

  1. 《TCP/IP网络编程》 尹圣雨

I/O复用

多进程服务器端的缺点

多进程服务器端,只要有客户端连接请求就会创建新进程。但创建进程时需要付出极大代价。这需要大量的运算和内存空间,由于每个进程都具有独立的内存空间,所以相互间的数据交换也要求采用相对复杂的方法

select函数

使用select函数时可以将多个文件描述符集中到一起统一监视(监视项称为事件):

  1. 是否存在套接字接收数据?
  2. 无需阻塞传输数据的套接字有哪些?
  3. 哪些套接字发生了异常?

select函数的调用方法:
(1)步骤一:

  1. 设置文件描述符
  2. 指定监视范围
  3. 设置超时

(2)步骤二:

  1. 调用select函数

(3)步骤三:

  1. 查看调用结果
设置文件描述符

利用select函数可以同时监视多个文件描述符。需要将要监视的文件描述符集按照监视项(接收、传输、异常)进行区分

使用fd_set数据变量存储各类被监视的文件描述符。fd_set数组是存有0和1的位数组,如果该位设置为1,则表示该文件描述符是监视对象

在fd_set变量中注册或更改值的操作都由宏完成:

  1. FD_ZERO(fd_set* fdset);将fd_set变量的所有位初始化为0

  2. FD_SET(int fd, fd_set* fdset):在参数fdset指向的变量中注册文件描述符fd的信息(将对应位置为1)

  3. FD_CLR(int fd, fd_set* fdset):从参数fdset指向的变量中消除文件描述符fd的信息(将对应位置为0)

  4. FD_ISSET(int fd, fd_set* fdset):若参数fdset指向的变量中包含文件描述符fd的信息,则返回真

设置监视范围

select函数介绍:

#include <sys/select.h>
#include <sys/time.h>

int select(int maxfd, fd_set* readset, fd_set* writeset, fd_set* exceptset, const struct timeval* timeout);

(1)maxfd
监视对象文件描述符数量
(2)readset
将所有关注“是否存在待读取数据”的文件描述符注册到fd_set型变量,并传递其地址值
(3)writeset
将所有关注“是否可传输无阻塞数据”的文件描述符注册到fd_set型变量,并传递其地址值
(4)exceptset
将所有关注“是否发生异常”的文件描述符注册到fd_set型变量,并传递其地址值
(5)timeout
调用select函数后,为防止陷入无限阻塞的状态,传递超时(time-out)信息
(6)返回值
发生错误时返回-1,超时返回时返回0。因发生关注的事件返回时,返回大于0的值,该值是发生事件的文件描述符数。根据3个fd_set型变量,分别向其注册文件描述符信息,并把变量的地址值传递到上述函数的第二到第四个参数

文件描述符的监视范围与第一个参数有关,实际上,select函数要求通过第一个参数传递监视对象文件描述符的数量。但每次新建文件描述符时,其值都会增1,故只需将最大的文件描述符值加1再传递到select函数即可。加1是因为文件描述符的值从0开始

设置超时

设置超时是为了防止select函数一直阻塞,只要过了指定时间,就可以返回。超时时间与最后一个参数有关,其中timeval结构体如下:

struct timeval
{
	long tv_sec;
	long tv_usec;
};

将秒数填入tv_sec成员,将微秒数填入tv_usec成员。如果不想设置超时,则传递NULL

调用select函数后查看结果

文件描述符的变化是指监视的文件描述符中发生了相应的监视事件。select函数调用完成后,向其传递的fd_set变量中将发生变化。原来为1的所有位均变为0,但发生变化的文件描述符对应位除外。因此,可以认为值仍为1的位置上的文件描述符发生了变化

select函数示例
#include <stdio.h>
#include <unistd.h>
#include <sys/time.h>
#include <sys/select.h>
#define BUF_SIZE 30

int main(int argc, char* argv[])
{
    fd_set reads, temps;
    int result, str_len;
    char buf[BUF_SIZE];
    struct timeval timeout;
    FD_ZERO(&reads);
    FD_SET(0, &reads);            // 将文件描述符0对应位设置为1

    while (1)
    {
        temps = reads;            // 为了记住初始值,必须经过这种复制过程,通用方法
        timeout.tv_sec = 5;       // 每次调用都初始化timeout
        timeout.tv_usec = 0;
        result = select(1, &temps, 0, 0, &timeout);
        if (result == -1)
        {
            puts("select() error!");
            break;
        }
        else if (result == 0)     // 没有输入,引发超时
        {
            puts("Time-out!");
        }
        else
        {
            if (FD_ISSET(0, &temps))    // 验证发送变化的文件描述符是否为标准输入
            {
                str_len = read(0, buf, BUF_SIZE);
                buf[str_len] = 0;
                printf("message from console: %s", buf);
            }
        }
    }
    return 0;
}

实现I/O复用服务器端

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <sys/select.h>

#define BUF_SIZE 100
void error_handling(char* buf);

int main(int argc, char* argv[])
{
    int serv_sock, clnt_sock;
    struct sockaddr_in serv_adr, clnt_adr;
    struct timeval timeout;
    fd_set reads, cpy_reads;

    socklen_t adr_sz;
    int fd_max, str_len, fd_num, i;
    char buf[BUF_SIZE];
    if (argc != 2)
    {
        printf("Usage : %s <port>\n", argv[0]);
        exit(1);
    }

    serv_sock = socket(PF_INET, SOCK_STREAM, 0);
    memset(&serv_adr, 0, sizeof(serv_adr));
    serv_adr.sin_family = AF_INET;
    serv_adr.sin_addr.s_addr = htonl(INADDR_ANY);
    serv_adr.sin_port = htons(atoi(argv[1]));

    if (bind(serv_sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr)) == -1)
    {
        error_handling("bind() error");
    }
    if (listen(serv_sock, 5) == -1)
    {
        error_handling("listen() error");
    }

    FD_ZERO(&reads);
    FD_SET(serv_sock, &reads);   // 客户端的连接请求同样通过传输数据完成。因此,服务器端中有接收数据,就意味着有新的连接请求
    fd_max = serv_sock;

    while (1)
    {
        cpy_reads = reads;
        timeout.tv_sec = 5;
        timeout.tv_usec = 5000;

        if ((fd_num = select(fd_max + 1, &cpy_reads, 0, 0, &timeout)) == -1)
        {
            break;
        }
        if (fd_num == 0)
        {
            continue;
        }

        for (i = 0; i < fd_max + 1; i++)
        {
            if (FD_ISSET(i, &cpy_reads))
            {
                if (i == serv_sock)  // 如果是服务器端套接字变化,说明是连接请求
                {
                    adr_sz = sizeof(clnt_adr);
                    clnt_sock = accept(serv_sock, (struct sockaddr*)&clnt_adr, &adr_sz);
                    FD_SET(clnt_sock, &reads); // 注册客户端套接字文件描述符为监视对象
                    if (fd_max < clnt_sock)
                    {
                        fd_max = clnt_sock;
                    }
                    printf("connected client: %d \n", clnt_sock);
                }
                else   // 要接收数据
                {
                    str_len = read(i, buf, BUF_SIZE);
                    if (str_len == 0)
                    {
                        FD_CLR(i, &reads);    // 接收到EOF,关闭套接字,并注销监视对象
                        close(i);
                        printf("closed client: %d \n", i);
                    }
                    else
                    {
                        write(i, buf, str_len); // 回声服务
                    }
                }
            }
        }
    }
    close(serv_sock);
    return 0;
}

void error_handling(char* message)
{
    fputs(message, stderr);
    fputc('\n', stderr);
}

基于Windows实现I/O复用服务器端

select函数

Windows的select函数所有参数与Linux相同,但前者的第一个参数是为了兼容性而添加的,没有意义

#include <WinSock2.h>

int select(int nfds, fd_set* readfds, fd_set* writefds, fd_set* excepfds, const struct timeval* timeout);

成功时返回0,失败时返回-1。其中timeval结构体使用typedef声明:

typedef struct timeval
{
	long tv_sec;
	long tv_usec;
}TIMEVAL;

fd_set结构体并非Linux下那样的位数组:

typedef struct fd_set
{
	u_int fd_count;
	SOCKET fd_array[FD_SETSIZE];
}fd_set;

fd_count用于套接字句柄数,fd_array用于保存套接字句柄。因为Windows的套接字句柄并非从0开始,而且句柄的整数值之间并无规律可循,需要直接保存句柄的数组和记录句柄数的变量

实现I/O复用服务器端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <WinSock2.h>

#define BUF_SIZE 1024
void ErrorHandling(char* message);

int main(int argc, char* argv[])
{
    WSADATA wsaData;
    SOCKET hServSock, hClntSock;
    SOCKADDR_IN servAdr, clntAdr;
    TIMEVAL timeout;
    fd_set reads, cpyReads;

    int adrSz;
    int strLen, fdNum, i;
    char buf[BUF_SIZE];

    if (argc != 2)
    {
        printf("Usage : %s <port>\n", argv[0]);
        exit(1);
    }
    if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
    {
        ErrorHandling("WSAStartup() error!");
    }

    hServSock = socket(PF_INET, SOCK_STREAM, 0);
    memset(&servAdr, 0, sizeof(servAdr));
    servAdr.sin_family = AF_INET;
    servAdr.sin_addr.s_addr = htonl(INADDR_ANY);
    servAdr.sin_port = htons(atoi(argv[1]));

    if (bind(hServSock, (SOCKADDR*)&servAdr, sizeof(servAdr)) == SOCKET_ERROR)
    {
        ErrorHandling("bind() error");
    }
    if (listen(hServSock, 5) == SOCKET_ERROR)
    {
        ErrorHandling("listen() error");
    }

    FD_ZERO(&reads);
    FD_SET(hServSock, &reads);

    while (1)
    {
        cpyReads = reads;
        timeout.tv_sec = 5;
        timeout.tv_usec = 5000;

        if ((fdNum = select(0, &cpyReads, 0, 0, &timeout)) == SOCKET_ERROR)
        {
            break;
        }

        if (fdNum == 0)
        {
            continue;
        }

        for (i = 0; i < reads.fd_count; i++)
        {
            if (FD_ISSET(reads.fd_array[i], &cpyReads))
            {
                if (reads.fd_array[i] == hServSock)
                {
                    adrSz = sizeof(clntAdr);
                    hClntSock = accept(hServSock, (SOCKADDR*)&clntAdr, &adrSz);
                    FD_SET(hClntSock, &reads);
                    printf("connected client: %d \n", hClntSock);
                }
                else
                {
                    strLen = recv(reads.fd_array[i], buf, BUF_SIZE - 1, 0);
                    if (strLen == 0)
                    {
                        FD_CLR(reads.fd_array[i], &reads);
                        closesocket(cpyReads.fd_array[i]);
                        printf("closed client: %d \n", cpyReads.fd_array[i]);
                    }
                    else
                    {
                        send(reads.fd_array[i], buf, strLen, 0);
                    }
                }
            }
        }
    }
    closesocket(hServSock);
    WSACleanup();
    return 0;
}

void ErrorHandling(char* message)
{
    fputs(message, stderr);
    fputc('\n', stderr);
    exit(1);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值