Java NIO和传统的IO的区别

278 阅读4分钟

小知识,大挑战!本文正在参与“程序员必备小知识”创作活动。

Java NIO和IO的区别。

IONIO
面向流(Stream Oriented)面向缓冲区(Buffer Oriented)
阻塞IO(Blocking IO)非阻塞IO(NonBlocking IO)
(无)选择器(Selectors)

1 面向流和面向缓冲区

Java IO 是面向流的,而Java NIO是面向缓冲区的。

在Java IO中读取数据和写入数据是面向流(Stream)的,这表示当我们从流中读取数据,写入数据时也将其写入流,流的含义在于没有缓存,就好像我们站在流水线前,所有的数据沿着流水线依次到达我们的面前,我们只能读取当前的数据(相当于我们拥有一个数据流的切面)。如果需要获取某个数据的前一项或后一项数据那就必须自己缓存数据,而不能直接从流中获取(因为面向流就意味着我们只有一个数据流的切面)。

而在Java NIO中数据的读写是面向缓冲区(Buffer)的,读取时可以将整块的数据读取到缓冲区中,在写入时则可以将整个缓冲区中的数据一起写入。这就好像是将流水线传输变成了卡车运送,面向流的数据读写只提供了一个数据流切面,而面向缓冲区的IO则使我们能够看到数据的上下文,也就是说在缓冲区中获取某项数据的前一项数据或者是后一项数据十分方便。这种便利是有代价的,因为我们必须管理好缓冲区,这包括不能让新的数据覆盖了缓冲区中还没有被处理的有用数据;将缓冲区中的数据正确的分块,分清哪些被处理过哪些还没有等等。

Java NIO的IO模型与很多IO的本质更加一致!磁盘IO读写就是数据块读写; TCP/IP协议传输的也是数据包而不是数据流。

2 阻塞和非阻塞

Java IO是阻塞的,如果在一次读写数据调用时数据还没有准备好,或者目前不可写,那么读写操作就会被阻塞直到数据准备好或目标可写为止,因为当用read去读取网络的数据时,是无法预知对方是否已经发送数据的。因此在收到数据之前,能做的只有等待,直到对方把数据发过来,或者等到网络超时。

Java NIO则是非阻塞的,每一次数据读写调用都会立即返回,并将目前可读(或可写)的内容从缓冲区中输出或者写入缓冲区,即使当前没有可用数据,调用仍然会立即返回并且不对缓冲区做任何操作。

这就好像去超市买东西,如果超市中没有需要的商品或者数量还不够,那么Java IO会一直等直到超市中需要的商品数量足够了就将所有需要的商品带回来,Java NIO则不同,不论超市中有多少需要的商品,它都会立即买下可以买到的所有需要的商品并返回,甚至是没有需要的商品也会立即返回。

阻塞IO会使得线程将大量的时间浪费在等待IO操作就绪上,这是非常不划算的,但是这种阻塞可以在数据可用时立即获取并处理数据,而非阻塞IO则在IO没有就绪时不会阻塞,但必须通过重复循环的调用来获取全部数据。

Java NIO 使用Selector实现单线程管理多个Channel,通过 select 调用,可以获取已经准备好的Channel并进行相应的处理。

3 Selector

NIO具有选择器Selector,选择器用于监听多个管道Channel的各种事件,使用传统的阻塞IO时我们可以方便的知道什么时候可以进行读写,因为它在可读或者可写之前一直是是阻塞的。

而使用非阻塞通道,读取不到数据会立即返回,因此我们需要一些方法来知道什么时候通道准备好了,Selector选择器正是为这个需要而诞生的。

这实际上属于IO多路复用的优点。

如有需要交流,或者文章有误,请直接留言。另外希望点赞、收藏、关注,我将不间断更新各种Java学习博客!