<TCP/IP网络编程> Chap12. I/O复用

理解select函数并实现服务器端

select函数调用过程:

步骤1
----------------
| 设置文件描述符 |
|  指定监视范围  |
|   设置超时    |
----------------
       ↓
步骤2
----------------
| 调用select函数 |
----------------
       ↓
步骤3
----------------
|  查看调用结果  |
----------------
#include <sys/select.h>
#include <sys/time.h>
/* 
 * @params
 *   maxfd: 监视对象文件描述符数量
 *   readset: 将所有关注”是否存在待读取数据“的文件描述符注册到fd_set型变量,并传递其地址值
 *   writeset: 将所有关注”是否可传输无阻塞数据“的文件描述符注册到fd_set型变量,并传递其地址值
 *   exceptset: 将所有关注”是否发生异常“的文件描述符注册到fd_set型变量,并传递其地址值
 *   timeout: select函数是阻塞的,为防止陷入无限阻塞状态,传递超时信息
 */
int select(int maxfd, fd_set *readset, fd_set *writeset, fd_set *exceptset, const struct timeval *timeout);    // 成功时返回发生关注事件的fd的个数(大于0),失败时返回-1,超时返回0

select函数调用完成后,fd_set变量会发生变化,除发生变化的文件描述符对应位不变,其余位均变为0。
示例:

# gcc select.c -o select
# ./select 
Time out!
Time out!
Hi~
Message feom console: Hi~

应用到回声服务器中(客户端可以直接用第十一章的):

# gcc echo_selectserv.c -o selserv
# ./selserv 9190
Time out!
Time out!
Time out!
Connected client: 4 
Time out!
Time out!
Connected client: 5 
Time out!
Closed client: 4 
Closed client: 5
# gcc echo_client.c -o eclient
# ./eclient 127.0.0.1 9190
Connected
Input message (Q to quit): first
Message from server: first
Input message (Q to quit): bye
Message from server: bye
Input message (Q to quit): q
# ./eclient 127.0.0.1 9190
Connected
Input message (Q to quit): second
Message from server: second
Input message (Q to quit): qingtian
Message from server: qingtian
Input message (Q to quit): Q


习题

  1. 请解释复用技术的通用含义,并说明何为I/O复用。
    复用技术是为了提高物理设备的效率,用最少的物理要素传递最多数据时使用的技术。I/O复用是通过一个进程向多个客户端提供服务。
  2. 多进程并发服务器的缺点有哪些?如何在I/O复用服务器端中弥补?
    每个进程都占用一份内存,数据交换也较复杂。I/O复用服务器通过一个进程向多个客户端提供服务,避免了这些问题。
  3. 复用服务器端需要select函数。下列关于select函数使用方法的描述错误的是?
    a. 调用select函数前需要集中I/O监视对象的文件描述符。
    b. 若已通过select函数注册为监视对象,则后续调用select函数时无需重复注册。
    c. 复用服务器端同一时间只能服务于1个客户端,因此,需要服务的客户端接入服务器端后只能等待。
    d. 与多进程服务器端不通,基于select的复用服务器端只需要1 个进程。因此,可以减少因创建进程产生的服务器端的负担。
    b。
  4. select函数的观察对象中应包含服务器端套接字(监听套接字),那么应将其包含到哪一类监听对象集合?请说明原因。
    “是否存在待读取数据”集合。为什么要加进这个集合?


我的问题

  1. 如果select返回后程序没有做处理,下一次再调用select函数,这些事件还在吗?


附录

[1] Github

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。