NIO(New I/O)是Java中的一种新输入输出库,它提供了一种更高效的文件读写方式。相比于传统的IO库,NIO具有以下特点:
非阻塞性:NIO支持非阻塞I/O操作,这意味着在执行I/O操作时,程序不会被阻塞,而是可以继续执行其他任务。
通道(Channel):NIO中的通道(Channel)是一个抽象概念,它可以表示文件、网络连接等I/O资源。通过通道,可以实现数据的读写和传输。
缓冲区(Buffer):NIO使用缓冲区来存储数据,这样可以提高数据传输的效率。缓冲区可以是字节缓冲区(ByteBuffer)或字符缓冲区(CharBuffer)。
选择器(Selector):NIO中的选择器(Selector)用于监控多个通道的状态变化,当某个通道有数据可读或可写时,可以选择器会通知应用程序。这样可以避免轮询多个通道,提高程序的性能。
下面是一个简单的NIO示例代码:
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Iterator;
public class NIOExample {
public static void main(String[] args) throws IOException {
// 创建选择器
Selector selector = Selector.open();
// 打开服务器套接字通道
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
serverSocketChannel.configureBlocking(false);
serverSocketChannel.bind(new InetSocketAddress("localhost", 8080));
// 注册到选择器,关注接受事件
serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);
while (true) {
// 选择一组键,其相应的通道已为 I/O 操作准备就绪
int readyChannels = selector.select();
if (readyChannels == 0) continue;
// 获取选择器选中的键集合
Iterator<SelectionKey> keyIterator = selector.selectedKeys().iterator();
while (keyIterator.hasNext()) {
SelectionKey key = keyIterator.next();
if (key.isAcceptable()) {
// 处理接受事件
SocketChannel socketChannel = serverSocketChannel.accept();
socketChannel.configureBlocking(false);
socketChannel.register(selector, SelectionKey.OP_READ);
} else if (key.isReadable()) {
// 处理读取事件
SocketChannel socketChannel = (SocketChannel) key.channel();
ByteBuffer buffer = ByteBuffer.allocate(1024);
int bytesRead = socketChannel.read(buffer);
if (bytesRead > 0) {
buffer.flip();
// 处理读取到的数据
while (buffer.hasRemaining()) {
System.out.print((char) buffer.get());
}
buffer.clear();
} else if (bytesRead < 0) {
// 客户端关闭连接
socketChannel.close();
}
}
// 移除已处理的键
keyIterator.remove();
}
}
}
}