是否要将侦听socket设置为非阻塞?
情况一: one thread one loop, 阻塞侦听socket一个线程
- 这种情况下, 其他客户端的连接分配到其他的网络线程中(一个线程一个epoll loop),客户端socket必须是非阻塞。
- 主线程阻塞接收客户端的连接(只有一个while(true)循环,没有挂载到epoll上),并分配给其他的网络线程。
- 缺点:
- 单独给阻塞的侦听socket分配一个线程,并没有挂载到epoll上,浪费线程资源。
情况二:基于情况一,将阻塞的侦听fd挂载到epoll上
- 当有客户端连接的时候,epoll_wait返回调用accept接收连接,这样accept就不会阻塞线程,提高了线程的使用率。
- 缺点:
- 每个epoll_wait的轮询只能接收一个客户端连接,一次有多个客户端连接的时候,也只会调用一次accept,存在效率问题,相当于一次epoll_wait一次连接。
- 但是在侦听fd的读事件触发的时候,用一个while循环接收连接,但是如果没有新连接的时候,accept会阻塞while循环,从而阻塞整个epoll loop。
情况三:将侦听fd设置为非阻塞的,挂载到epoll上
- 可以在侦听fd有事件发生的时候,就可以通过while循环来接收接连,如果没有新连接发生,就可以将通过错误码EWOULDBOLCK返回。
- 优点:
- 可以一个epoll_wait的时候,接收多个连接,并且可以自由设置一次最大可以接收的连接数量。
TIP:
- 对于one thread one loop的模型,主线程非阻塞accept挂载到epoll上,对于长连接的情况,也是比较浪费线程资源,但是对于短链接,会非常适应。
- 对于长连接的话,可以选择主线程的loop中的epoll也可以挂载客户端的fd。
结论:
不一定要将侦听fd设置为非阻塞,不用IO复用管理即可,单独一个线程accept循环阻塞等待客户端连接。