Java NIO概述
NIO(New IO),自jdk1.4发布的特性,是一种可以替代IO API的特性,这以为着可以替代标准的Java IO与Java Networking的API。Java NIO相比于标准的IO API接口有这不同的工作机制。
- 通道和缓冲(Channels and Buffers)
- 标准IO中使用字节流与字符流;
- NIO中使用通道和缓冲,数据是始终是从通道读取到缓冲,或者从缓冲中写入到通道。
- 非阻塞IO
- Java NIO可以使用非阻塞IO。例如,一个线程可以请求一个通道将数据读入到缓冲区中。通道读取数据缓冲区时,线程可以做其他的事情。一旦数据读取到缓冲区,线程可以继续处理它。这同样适用于将数据从缓冲区写入到通道。
- 选择器(Selectors)
- Java NIO包含“选择器”的概念。选择器是一个对象,可以监视多个渠道事件(如:连接打开,数据接受等等)。因此,一个线程可以监视多个渠道数据。
NIO核心
- Java NIO包括以下核心组件:
- Channels(通道)
- Buffers(缓冲)
- Selectors(选择器)
Java NIO有更多的类和组件,但Channels,Buffers和Selectors组成了核心API。其余的组件,如Pipe和FileLock仅仅是用于结合使用这三个核心组件的实用工具类。
Channels and Buffers
通常,NIO中所有的IO始于一个通道。通道有点像流。从通道中,数据可以读取到缓冲区。数据也可以从缓冲区写进一个通道。图示: ![这里写图片描述](https://img-blog.csdn.net/20180911154330235?watermark/2/text/aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L20wXzM3MTEzNTM5/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70)Channel
- Channel的主要实现类:
- FileChannel:读取文件与写入文件
- DatagramChannel:通过UDP读写数据
- SocketChannel:通过TCP读写数据
- ServerSocketChannel:监听TCP连接,类似web服务器。每次输入的连接都会创建一个SocketChannel
如上,这些通道覆盖了UDP+TCP的网络IO与文件IO
- 例子
RandomAccessFile aFile = new RandomAccessFile("data/nio-data.txt", "rw");
FileChannel inChannel = aFile.getChannel();
ByteBuffer buf = ByteBuffer.allocate(48);
int bytesRead = inChannel.read(buf);
while (bytesRead != -1) {
System.out.println("Read " + bytesRead);
buf.flip();
while(buf.hasRemaining()){
System.out.print((char) buf.get());
}
buf.clear();
bytesRead = inChannel.read(buf);
}
aFile.close();
Buffer
- NIO中Buffer的主要实现类:
- ByteBuffer
- CharBuffer
- DoubleBuffer
- FloatBuffer
- IntBuffer
- LongBuffer
- ShortBuffer
这些是Buffer的覆盖基本数据类型,您可以通过IO发送这几种类型:byte,short,int,long,float,double。
另外,Java NIO中也有内存映射文件一起使用的MappedByteBuffer。
Selectors
Selectors允许一个线程处理多个Channels。如果应用程序有许多连接(Channels)打开但每个连接都只有低传输量的时候,这非常敏捷。例如,聊天服务器的情形。下面是一个例子的一个线程使用一个Selector来处理3个Channel:
使用注册通道的选择器吧。你可以定义它为select()方法。该方法在有一个等待已经注册的Channel的事件之前会一直阻塞。一旦方法返回,线程可以处理这些事件。比如说,接收到事件的连接,数据接受等等。