之前学习过使用多进程创建服务器,现在介绍建立多进程服务器的另一种方法:I/O 复用技术。
什么是复用?
拿一个来看一下,下面是我们小时候经常玩的纸杯通话,加入有 3 个人对话的话,我们就需要 6 个杯子来实现这个对话系统。

而我们如果引入复用技术,这个时候就可以在只有 3 个杯子的情况下,1 个人对话而另外 2 个人同时听见,这个时候便会节省资源。

现在考虑一下上述复用系统相比于前一种的优点和缺点:
优点:
- 减少连线长度
- 减少纸杯个数
缺点:
- 保密性不强:不能实现 3 人中的 2 人单独通话
- 不能同时说话,因为现在说话和听都是一根线,所以只能一个人说完,另一个人说
在《自顶向下-计算机网络一书中》,第 1 章介绍时分电路和频分电路,而这个就是生活中最常用到的复用技术。
复用技术在服务器端的应用
纸杯电话系统引入复用技术后,可以减少纸杯书和连线长度。同样,服务器端引入复用技术后可以减少所需进程数,下面是两种模型的对比。

相对比与多进程服务器,我们必须先让客户端请求一下服务,然后建立连接。拿一个例子来看,就像 1 个老师回答 10 个同学的问题,教师必须先确定是否有学生举手,然后对举手的学生回答问题。
Linux 实现 I/O 复用服务器端
下面是通过 select 函数实现 I/O 复用服务器端,之前已经给出了 select 函数的所有说明,下面只需要利用 select 函数实现服务器端。下面是基于 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 100void error_handling(char *buf);int main(int argc, char *argv[]) {// 1.检查输入格式if (argc != 2) {printf("Usage %s <port> \n", argv[0]);exit(1);}// 2.设定套接字int serv_sock, clnt_sock;struct sockaddr_in serv_adr, clnt_adr;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]));// 3.绑定套接字if (bind(serv_sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr)) == -1)error_handling("bind() error");// 4.进入监听状态if (listen(serv_sock, 5) == -1)error_handling("listen() error");// 5.初始化 fd_set 数组fd_set reads, cpy_reads;FD_ZERO(&reads);FD_SET(serv_sock, &reads); // 监听文件描述符 serv_sock,即套接字fd_max = serv_sock;struct timeval timeout;socklen_t adr_sz;int fd_max, str_len, fd_num, i;char buf[BUF_SIZE];while(1) {cpy_reads = reads; // 注意:需要保留一个最初的 reads,这个时候只监听 hServSock,而没有客户端套接字timeout.tv_sec = 5;timeout.tv_usec = 5000;// 6.select 调用出错if ( (fd_num=select(fd_max+1, &cpy_reads, 0, 0, &timeout)) == -1)break;// 7.select 超时if (fd_num==0)continue;for (i=0; i<fd_max+1; i++) {// 8.循环找到发生变化的文件描述符if (FD_ISSET(i, &cpy_reads)) {// 8.1.如果发生变化的套接字是监听的服务器端套接字 serv_sock,客户端请求建立连接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 (if_max < clnt_sock)fd_max = clnt_sock;printf("connected client: %d \n", clnt_sock);}// 8.2.发生变化的套接字并非服务器端套接字,而是客户端套接字 clnt_sock。即有要接// 收的数据执行else语句。但此时需要确认接收的数据是字符串还是代表断开连接的EOFelse {str_len = read(i, buf, BUF_SIZE);if (str_len == 0) { // 接收的数据为EOF时关闭套接字,并删除reads相应信息FD_CLR(i, &reads);close(i);printf("closed client: %d \n", i);}else { // 收到客户端字符串,则将该字符串再发给客户端,回声write(i, buf, str_len);}}}}}close (serv_sock);return 0;}void error_handling(char *buf) {fputs(buf, stderr);fputc('\n', stderr);exit(1);}
配合上之前学习的多进程并发服务器和今天学习的 I/O 服用服务器,可以看到并发服务器最终实现的效果如下图所示:

其中,
- serv_sock 配合 accept 函数与客户端连接,分配和该客户端的连接通道 clnt_sock
- 使用 clnt_sock 通信
多进程并发服务器为每个 clnt_sock 分配一个子进程来通信,而 I/O 复用服务器则是通过父进程循环判断子进程描述符从而实现与客户端通信。
Windows 实现 I/O 复用服务器端
Windows 同样提供 select 函数,而且所有参数与 Linux 的 select 函数完全相同。只不过 Windows 平台 select 函数的第一个参数是为了保持与(包括 Linux 的)UNIX 系列操作系统的兼容性而添加的,并没有特殊意义。
select 函数
select 函数原型如下
#include <winsock2.h>// select函数调用成功时返回0,失败时返回 -1int select(int nfds,fd_set *readfds, fd_set *writefds, fd_set *excepfds,const struct timeval *timeout);
这个函数的返回值、参数顺序及含义与之前 Linux 的 select 函数完全相同,这里省略。下面给出 timeval 结构体定义。
typedef struct timeval {long tv_sec; // secondslong tv_usec; // microseconds} TIMEVAL;
基本结构与之前 Linux 中定义的相同,但是 Windows 这里使用的是 typedef 声明。接下来观察 fd_set 结构体,Windows 中实现时需要注意的地方就在于此,因为 Windows 的 fd_set 并不像 Linux 那样使用了位数组。
typedef struct fd_set {u_int fd_count; // 用于套接字句柄数,即有多少个套接字SOCKET fd_array[FD_SETSIZE]; // 保存套接字句柄,套接字有哪些} fd_set;
可以看到,上述的 fd_set 是由成员 fd_count 和 fd_array 构成的,这里我们思考一下为什么 Linux 和 Windows 的 fd_set 有这样的区别。
- Linux 的文件描述符从 0 开始递增,套接字取得是值最小的文件描述符,所以很容易找到文件描述符数量和套接字关系
- Windows 的套接字句柄并不是从 0 开始,所以句柄的值之间是没有什么规律的,所以需要使用 fd_count 记录套接字的数量,并且使用 fd_array 记录套接字。
微软为了保证兼容性,保留了处理 fd_set 结构体的 FD_XXX 型的 4 个宏,其功能、名称及使用方法和 Linux 完全相同。
基于 Windows 实现 I/O 复用服务器端
下面是 Windows 的 I/O 复用服务器端实现代码
#include <stdio.h>#include <stdlib.h>#include <string.h>#include <winsock2.h>#define BUF_SIZE 1024void ErrorHandling(char *message);int main(int argc, char *argv[]) {// 1.检验输入if (argc != 2) {printf("Usage : %d <port> \n", argv[0]);exit(1);}// 2.指明 Winsock 版本WSADATA wsaData;if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) // 2.2 版本ErrorHandling("WSAStartup() error!");// 3.初始化套接字描述符SOCKET hServSock, hClntSock;SOCKADDR_IN servAdr, clntAdr;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]);)// 4.绑定套接字if (bind(hServSock, (SOCKADDR*)&servAdr, sizeof(servAdr)) == SOCKET_ERROR)ErrorHandling("bind() error");// 5.监听状态if (listen(hServSock, 5) == SOCKET_ERROR)ErrorHandling("listen() error");// 6.初始化 fd_set 数组fd_set reads, cpyReads;FD_ZERO(&reads);FD_SET(hServSock, &reads); // 监听 hServSock 套接字,查看是否有客户端连接TIMEVAL timeout;int adrSz;int strLen, fdNum, i;char buf[BUF_SIZE];while(1) {cpyReads = reads; // 注意:需要保留一个最初的 reads,这个时候只监听 hServSock,而没有客户端套接字timeout.tv_sec = 5;timeout.tv_usec = 5000;if ( (fdNum=select(0, &cpyReads, 0, 0, &timeout)) == SOCKET_ERROR) // select 函数出错break;if (fdNum == 0)continue;for (i=0; i<reads.fd_count; i++) {// 7.如果套接字发生了变化if (FD_ISSET(reads.fd_array[i], &cpyReads)) {// 7.1.如果是服务器套接字,则说明有新的客户端请求连接if (reads.fd_array[i] == hServSock) {adrSz = sizeof(clntAdr);hClntSock = accept(hServSock, (SOCKADDR*)&clntAdr, &adrSz); // 为该客户端分配套接字 hClntSockFD_SET(hClntSock, &reads); // 监听该套接字printf("connected client: %d \n", hClntSock);}// 7.2如果是某个客户端套接字 hClntSock,则处理请求else {streLen = recv(reads.fd_array[i], buf, BUF_SIZE-1, 0);if (strLen == 0) { // 7.2.1.对于EOFFD_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); // 7.2.2.回声服务器}}}}}closesocket(hServSock);WSACleanup();return 0;}void ErrorHandling(char *message) {fputs(message, stderr);fputc('\n', stderr);exit(1);}
