TCP的并发处理select

前面我们介绍的全都是一个客户端和一个服务端的情况,只适用于简单的应用场景,但是在实际的开发工作中,经常会用到一个服务器连接多个客户端的情况,如多个手机连接网关或者多个WIFI设备连接到网关等。

在Linux上处理Socket并发有好几种方式,我们着重介绍select和epoll,前者是可以跨平台的c函数,在Windows和Linux上都可以使用,后者是Linux平台特有的处理并发的函数。 本节先介绍select的使用。

select的图示:

每次调用select时,都会将描述集合fd从用户态拷贝到内核态,并为其注册回调,然后调用poll方法遍历这些fd,看看是否有描述符就绪,如果就绪了就给这个fd_set赋值,用户根据这个fd_set判断是哪个文件fd发生了变化,如果fd比较大,这部分开销就会变大,因为select每次都会操作所有的fd,select最大支持1024个描述符。

服务端程序如下:

有这么几个注意点:

1. select的第一个参数是监控fd set中的最大描述符加一,所以在接收到客户端请求后需要对比这个值,保持它是最大的。

2. select的监控集合fdSelect在每次调用select函数后会被清空,因此需要使用fdAll来保存fd的集合。

3. 程序中可连接的客户端最大数量为5,当达到这个值后应该将服务器的SocketFd清出fd集合,否则客户端的连接会使select不断返回,影响性能。

4. for循环后不要忘记break跳出循环,否则程序再次运行accept或者recv都会使线程挂起,select再没有机会运行。

5. 例程select中不仅做了接收客户端连接部分,并且接收了客户端的数据,但是不要在这里面处理数据,否则可能导致下次数据接收失败。

6. 多客户端通信时,可以只使用select做接收处理,然后配合非阻塞Socket进行通信,避免多线程的通信互斥问题。

7. select可以设置为非阻塞的,即设置成select(iSelectFd + 1, &fdSelect, NULL, NULL, &tv)模式

最后更新于

这有帮助吗?