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