多路复用 IO 模型是目前使用得比较多的模型。Java NIO 实际上就是多路复用 IO。


在多路复用 IO 模型中,会有一个线程不断去轮询多个 socket 的状态,只有当


socket 真正有读写事件时,才真正调用实际的 IO 读写操作。因为在多路复用 IO 模型中,


只需要使用一个线程就可以管理多个 socket,系统不需要建立新的进程或者线程,也不必


维护这些线程和进程,并且只有在真正有 socket 读写事件进行时,才会使用 IO 资源,所


以它大大减少了资源占用。



在 Java NIO 中,是通过 selector.select()去查询每个通道是否有到达事件,如果没


有事件,则一直阻塞在那里,因此这种方式会导致用户线程的阻塞。



也许有朋友会说,我可以采用 多线程+ 阻塞 IO 达到类似的效果,但是由于在多线程


+ 阻塞 IO 中,每个 socket 对应一个线程,这样会造成很大的资源占用,并且尤其是对于


长连接来说,线程的资源一直不会释放,如果后面陆续有很多连接的话,就会造成性能上的


瓶颈。



而多路复用 IO 模式,通过一个线程就可以管理多个 socket,只有当 socket 真正有读


写事件发生才会占用资源来进行实际的读写操作。因此,多路复用 IO 比较适合连接数比较


多的情况。



另外多路复用 IO 为何比非阻塞 IO 模型的效率高是因为在非阻塞 IO 中,不断地询问


socket 状态时通过用户线程去进行的,而在多路复用 IO 中,轮询每个 socket 状态是内核


在进行的,这个效率要比用户线程要高的多。



不过要注意的是,多路复用 IO 模型是通过轮询的方式来检测是否有事件到达,并且对


到达的事件逐一进行响应。因此对于多路复用 IO 模型来说,一旦事件响应体很大,那么就


会导致后续的事件迟迟得不到处理,并且会影响新的事件轮询