NIO基础
Buffer
Buffer工作原理:capacity - 缓冲区容量,缓冲区满了以后,必须清空后才能继续写入数据。position - 写数据到缓冲区时,position代表写入数据的当前位置,初始值为0,当一个字节数据写入到缓冲区后,position会移动到写一个可插入数据的缓冲区单元;当从缓冲区中读取数据时,position表示读取数据的当前位置,调用flip方法切换到读模式时,position会被重置为零。limit - 写数据时表示最多可以写入多少个数据;读数据时,表示还有多少个数据可读取。
//IntBufferDemo.java
import java.nio.IntBuffer;
public class IntBufferDemo {
public static void main(String[] args) {
//每个Buffer类都有一个allocate方法
IntBuffer buffer = IntBuffer.allocate(8);
for (int i = 0; i < buffer.capacity(); i++) {
int j = 2 * (i+1);
//可以使用通道或者put方法写入数据到缓冲区
buffer.put(j);
}
//clear方法清空缓冲区,compact方法只清除已读取过的数据
//mark方法可以标记缓冲区中的一个特定position,然后可以通过reset恢复到这个位置
//rewind方法是把position设置为0,可以重读缓冲区中的数据
//flip方法把缓冲区从写模式切换到读模式,把limit设置成position的当前值,然后把position清零
buffer.flip();
while(buffer.hasRemaining()) {
//从缓冲区读取数据,可以使用通道读取或者使用get方法
System.out.println(buffer.get() + ",");
}
}
}
//BufferDemo1.java
import java.io.RandomAccessFile;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
public class BufferDemo1 {
public static void main(String[] args) throws Exception {
RandomAccessFile file = new RandomAccessFile("01.txt", "rw");
//文件通道从缓冲区读取数据
FileChannel channel = file.getChannel();
ByteBuffer buffer = ByteBuffer.allocate(1024);
int bytesRead = channel.read(buffer);
while(bytesRead != -1){
buffer.flip();
while(buffer.hasRemaining()){
System.out.print((char)buffer.get());
}
buffer.clear();
bytesRead = channel.read(buffer);
}
file.close();
}
}
//BufferDemo2.java
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.ByteBuffer;
import java.nio.IntBuffer;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
public class BufferDemo2 {
public static void main(String[] args) throws IOException {
ByteBuffer buffer = ByteBuffer.allocate(10);
for(int i=0;i<buffer.capacity();i++){
buffer.put((byte)i);
}
//创建子缓冲区
buffer.position(3);
buffer.limit(7);
ByteBuffer sliceBuffer = buffer.slice();
for(int i=0;i<sliceBuffer.capacity();i++){
byte b = sliceBuffer.get(i);
b *= 10;
sliceBuffer.put(i, b);
}
buffer.position(0);
buffer.limit(buffer.capacity());
while(buffer.hasRemaining()){
System.out.println(buffer.get());
}
//创建只读缓冲区,只读缓冲区随着原缓冲区发生变化
IntBuffer intBuffer = IntBuffer.allocate(10);
for(int i=0;i<intBuffer.capacity();i++){
intBuffer.put(i);
}
IntBuffer readOnlyBuffer = intBuffer.asReadOnlyBuffer();
for(int i=0;i<intBuffer.capacity();i++){
int b = intBuffer.get(i);
b *= 10;
intBuffer.put(i, b);
}
readOnlyBuffer.position(0);
readOnlyBuffer.limit(intBuffer.capacity());
while(readOnlyBuffer.hasRemaining()){
System.out.println(readOnlyBuffer.get());
}
//直接缓冲区
//ByteBuffer directBuffer = ByteBuffer.allocateDirect(1024);
//内存映射文件I/O
RandomAccessFile file = new RandomAccessFile("02.txt", "rw");
FileChannel fileChannel = file.getChannel();
MappedByteBuffer mbb =
fileChannel.map(FileChannel.MapMode.READ_WRITE, 0, 1024);
mbb.put(0, (byte)97);
mbb.put(1023, (byte)122);
file.close();
}
}
- 使用多线程的方式实现
import java.io.*;
import java.nio.ByteBuffer;
import java.nio.IntBuffer;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
import java.security.SecureRandom;
import java.util.Arrays;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class NioBufferInstance {
public static void main(String[] args) {
ExecutorService service = Executors.newFixedThreadPool(10);
//获取流的位置
service.submit(() -> {
IntBuffer buffer = IntBuffer.allocate(10);
System.out.println("The capacity of ints buffer:" + buffer.capacity());
Arrays.asList(0, 1, 2, 3, 4)
.forEach(p -> buffer.put(new SecureRandom().nextInt(20)));
System.out.println("before flip limit:" + buffer.limit());
buffer.flip();
System.out.println("after flip limit:" + buffer.limit());
while(buffer.hasRemaining()){
System.out.println("position:" + buffer.position());
System.out.println("limit:" + buffer.limit());
System.out.println("capacity:" + buffer.capacity());
System.out.println(buffer.get());
}
});
//文件通道读写
service.submit(() -> {
try {
FileOutputStream os = new FileOutputStream("output.txt");
FileInputStream is = new FileInputStream("input.txt");
FileChannel inputChannel = is.getChannel();
FileChannel outputChannel = os.getChannel();
ByteBuffer buffer = ByteBuffer.allocate(4);
while(true){
buffer.clear();
int read = inputChannel.read(buffer);
System.out.println("read:" + read);
if(read == -1){
break;
}
buffer.flip();
outputChannel.write(buffer);
}
inputChannel.close();
outputChannel.close();
} catch (IOException e) {
e.printStackTrace();
}
});
//随机存取,使用堆外内存
service.submit(() -> {
try {
RandomAccessFile file = new RandomAccessFile("test.txt", "rw");
FileChannel channel = file.getChannel();
MappedByteBuffer buffer =
channel.map(FileChannel.MapMode.READ_WRITE, 0, 5);
buffer.put(0, (byte) 'a');
buffer.put(3, (byte) 'b');
file.close();
} catch (IOException e) {
e.printStackTrace();
}
});
//allocate wrap HeapByteBuffer 在JVM堆中创建缓冲区
//directBuffer DirectByteBuffer 在堆外创建缓冲区,直接与IO设备交互,零拷贝
service.submit(() -> {
ByteBuffer buffer = ByteBuffer.allocate(64);
buffer.putInt(15);
buffer.putLong(500000000L);
buffer.putChar('你');
buffer.putShort((short) 2);
buffer.putChar('我');
buffer.flip();
System.out.println(buffer.getInt() + ","
+ buffer.getLong() + "," + buffer.getChar()
+ ',' + buffer.getShort() + "," + buffer.getChar());
//重置position、limit后重新赋值
buffer.clear();
for(int i=0; i<buffer.capacity(); i++){
buffer.put((byte) i);
}
//指定position和limit
buffer.position(2);
buffer.limit(6);
//slice创建一个新的ByteBuffer,新的ByteBuffer操作的结果影响原ByteBuffer
ByteBuffer sliceBuffer = buffer.slice();
for(int i=0;i<sliceBuffer.capacity();i++){
byte b = sliceBuffer.get(i);
b *= 2;
sliceBuffer.put(i, b);
}
buffer.position(0);
buffer.limit(buffer.capacity());
while(buffer.hasRemaining()){
System.out.print(buffer.get() + "\t");
}
System.out.println();
buffer.clear();
System.out.println(buffer.getClass());
ByteBuffer readOnlyBuffer = buffer.asReadOnlyBuffer();
System.out.println(readOnlyBuffer.getClass());
});
//通道写
service.submit(() -> {
try {
FileOutputStream fos = new FileOutputStream("NioChannelDemo.txt");
FileChannel channel = fos.getChannel();
ByteBuffer buffer = ByteBuffer.allocate(512);
byte[] message = "Hello world, welcome!".getBytes();
for(byte b:message){
buffer.put(b);
}
buffer.flip();
channel.write(buffer);
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
});
//通道读
service.submit(() -> {
try {
FileInputStream fis = new FileInputStream("NioChannelDemo.txt");
FileChannel fileChannel = fis.getChannel();
ByteBuffer byteBuffer = ByteBuffer.allocate(512);
fileChannel.read(byteBuffer);
byteBuffer.flip();
while(byteBuffer.remaining() > 0){
byte b = byteBuffer.get();
System.out.println("Character:" + (char)b);
}
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
});
}
}
Channel
SocketChannel是用来连接到TCP网络套接字的通道,主要用来处理网络IO通道,基于TCP传输,实现了可选择通道,可以被多路复用。对于已经存在的套接字不能创建SocketChannel。没有进行连接的SocketChannel执行IO操作时会抛异常。SocketChannel的open方法创建的通道没有进行网络级联,需要使用connect方法连接到指定地址。SocketChannel支持异步关闭、设定参数。ServerSocketChannel是一个基于通道的socket监听器,调用accept方法返回SocketChannel。
//SocketChannelDemo.java
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;
public class SocketChannelDemo {
public static void main(String[] args) throws IOException {
//创建SocketChannel
SocketChannel socketChannel =
SocketChannel.open(new InetSocketAddress("www.baidu.com", 80));
socketChannel.configureBlocking(false);
ByteBuffer buffer = ByteBuffer.allocate(16);
socketChannel.read(buffer);
socketChannel.close();
System.out.println("read over");
}
}
//ServerSocketChannelDemo.java
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
public class ServerSocketChannelDemo {
public static void main(String[] args) throws IOException {
int port = 8888;
//创建Buffer
ByteBuffer buffer = ByteBuffer.wrap("hello world".getBytes());
//打开ServerSocketChannel,绑定端口号
ServerSocketChannel ssc = ServerSocketChannel.open();
ssc.bind(new InetSocketAddress(port));
//设置非阻塞模式
ssc.configureBlocking(false);
//监听新连接
for(;;){
System.out.println("Waiting for connecting...");
SocketChannel sc = ssc.accept();
if(sc == null){
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}else{
System.out.println("Incoming connection from:"
+ sc.socket().getRemoteSocketAddress());
buffer.rewind();
sc.write(buffer);
sc.close();
}
}
}
}
//PipeDemo.java
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.Pipe;
//管道是两个线程之间的单项数据连接,数据会被写到sink通道,从source通道读取
public class PipeDemo {
public static void main(String[] args) throws IOException {
//获取通道
Pipe pipe = Pipe.open();
Pipe.SinkChannel sinkChannel = pipe.sink();
ByteBuffer buffer = ByteBuffer.allocate(1024);
buffer.put("hello".getBytes());
buffer.flip();
sinkChannel.write(buffer);
Pipe.SourceChannel sourceChannel = pipe.source();
buffer.flip();
int len = sourceChannel.read(buffer);
System.out.println(new String(buffer.array(), 0,len));
sourceChannel.close();
sinkChannel.close();
}
}
//DatagramChannelDemo.java
import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.DatagramChannel;
//DatagramChannel是无连接的
public class DatagramChannelDemo {
public static void main(String[] args) throws IOException {
System.out.println("程序启动...");
}
private void sendDatagram() throws IOException {
DatagramChannel sendChannel = DatagramChannel.open();
InetSocketAddress sendAddress = new InetSocketAddress("localhost", 9999);
while(true){
ByteBuffer buffer = ByteBuffer.wrap("Hello World".getBytes());
sendChannel.send(buffer, sendAddress);
System.out.println("已经发送成功");
}
}
private void receiveDatagram() throws IOException {
DatagramChannel receiveChannel = DatagramChannel.open();
InetSocketAddress receiveAddress = new InetSocketAddress(9999);
receiveChannel.bind(receiveAddress);
ByteBuffer receiveBuffer = ByteBuffer.allocate(1024);
while(true){
receiveBuffer.clear();
SocketAddress socketAddress = receiveChannel.receive(receiveBuffer);
receiveBuffer.flip();
System.out.println(socketAddress.toString());
System.out.println(receiveBuffer);
}
}
}
//FileChannelDemo1.java
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
/**
* 通过FileChannel将数据写入缓冲区
*/
public class FileChannelDemo1 {
public static void main(String[] args) throws IOException {
//创建FileChannel
RandomAccessFile file =
new RandomAccessFile("C:\\Mermaid使用.md", "rw");
FileChannel channel = file.getChannel();
//创建Buffer
ByteBuffer buffer = ByteBuffer.allocate(1024);
//读取数据到buffer中
int bytesRead = channel.read(buffer);
while(bytesRead != -1){
System.out.println("读取了:" + bytesRead);
//读写模式转换
buffer.flip();
while(buffer.hasRemaining()){
System.out.println((char)buffer.get());
}
//清除缓冲区内容
buffer.clear();
bytesRead = channel.read(buffer);
}
file.close();
System.out.println("程序结束");
}
}
//FileChannelDemo2.java
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
/**
* 通过FileChannel从缓冲区读取数据
*/
public class FileChannelDemo2 {
public static void main(String[] args) throws IOException {
RandomAccessFile file = new RandomAccessFile("C:\\LICCD.txt", "rw");
FileChannel channel = file.getChannel();
ByteBuffer buffer = ByteBuffer.allocate(1024);
String data = "Test 12345";
buffer.clear();
buffer.put(data.getBytes());
buffer.flip();
//需要循环写入
while(buffer.hasRemaining()){
channel.write(buffer);
}
channel.close();
}
}
//FileChannelDemo3.java
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.channels.FileChannel;
/**
* 通道之间数据传输
*/
public class FileChannelDemo3 {
public static void main(String[] args) throws IOException {
RandomAccessFile aFile = new RandomAccessFile("C:\\01.txt", "rw");
RandomAccessFile bFile = new RandomAccessFile("C:\\02.txt", "rw");
FileChannel fromChannel = aFile.getChannel();
FileChannel toChannel = bFile.getChannel();
long position = 0;
long size = fromChannel.size();
//position() - 返回此通道文件位置, position(long) - 设置此通道的文件长度
//transferFrom - 把数据从源通道传输到FileChannel中;
//transferTo - 将数据从FileChannel传输到其他的FileChannel中
toChannel.transferFrom(fromChannel, position, size);
//fromChannel.transferTo(position, size, toChannel);
aFile.close();
bFile.close();
}
}
//FileLockDemo.java
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLock;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;
//文件锁的使用
public class FileLockDemo {
public static void main(String[] args) throws IOException {
String input = "Hello World!";
ByteBuffer buffer = ByteBuffer.wrap(input.getBytes());
String filePath = "01.txt";
Path path = Paths.get(filePath);
FileChannel channel = FileChannel.open(path, StandardOpenOption.WRITE,
StandardOpenOption.APPEND);
channel.position(channel.size() - 1);
FileLock lock = channel.lock();
System.out.println("是否共享锁:" + lock.isShared());
channel.write(buffer);
channel.close();
FileReader fileReader = new FileReader(filePath);
BufferedReader bufferedReader = new BufferedReader(fileReader);
String line = bufferedReader.readLine();
System.out.println("读取出内容:");
while(line != null){
System.out.println(line);
line = bufferedReader.readLine();
}
fileReader.close();
bufferedReader.close();
}
}
//AsyncFileChannelDemo.java
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousFileChannel;
import java.nio.channels.CompletionHandler;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;
/**
* 文件通道异步读取
*/
public class AsyncFileChannelDemo {
public static void main(String[] args) throws IOException {
Path path = Paths.get("01.txt");
AsynchronousFileChannel fileChannel =
AsynchronousFileChannel.open(path, StandardOpenOption.READ);
ByteBuffer buffer = ByteBuffer.allocate(1024);
// 使用Future处理结果集
// Future<Integer> future = fileChannel.read(buffer, 0);
// //判断是否完成
// while(!future.isDone());
// buffer.flip();
// while(buffer.hasRemaining()){
// System.out.print((char)buffer.get());
// }
// buffer.clear();
//使用CompletionHandler接口处理结果
fileChannel.read(buffer, 0, buffer,
new CompletionHandler<Integer, ByteBuffer>() {
@Override
public void completed(Integer result, ByteBuffer attachment) {
attachment.flip();
byte[] data = new byte[attachment.limit()];
attachment.get(data);
System.out.print(new String(data));
attachment.clear();
}
@Override
public void failed(Throwable exc, ByteBuffer attachment) {
System.out.println(exc.getMessage());
}
});
}
}
Selector
Selector选择器也叫做多路复用器,用于检查一个或多个通道的状态是否可读写。使用Selector的好处在于使用更少的线程处理通道,避免多线程上下文切换带来的开销。不是所有的Channel都可以被Selector复用的,继承可选择通道SelectableChannel就可以复用。一个通道可以被注册到多个选择器上,而每个选择器只能被注册一次。
//SelectorDemo.java
public class SelectorDemo {
public static void main(String[] args) throws IOException {
//与Selector多路复用器一起使用时,Channel必须是非阻塞的
Selector selector = Selector.open();
ServerSocketChannel ssc = ServerSocketChannel.open();
ssc.configureBlocking(false);
ssc.bind(new InetSocketAddress(8000));
ssc.register(selector, SelectionKey.OP_ACCEPT);
Set<SelectionKey> selectionKeys = selector.selectedKeys();
Iterator<SelectionKey> iterator = selectionKeys.iterator();
while(iterator.hasNext()) {
SelectionKey key = iterator.next();
if (key.isAcceptable()) {
System.out.println("accept");
} else if (key.isConnectable()) {
System.out.println("connect");
} else if (key.isReadable()) {
System.out.println("read");
} else if (key.isWritable()) {
System.out.println("write");
}
iterator.remove();
}
}
}
//SelectorTest.java
public class SelectorTest {
@Test
public void send() throws IOException {
//获取通道、绑定主机和端口,切换到非阻塞模式,
//创建Buffer写入数据,模式切换,写入通道,关闭
SocketChannel sc = SocketChannel.open(new InetSocketAddress(8888));
sc.configureBlocking(false);
ByteBuffer buffer = ByteBuffer.allocate(1024);
buffer.put(new Date().toString().getBytes());
buffer.flip();
sc.write(buffer);
buffer.close();
}
@Test
public void recevie() throws IOException {
ServerSocketChannel ssc = ServerSocketChannel.open();
ssc.bind(new InetSocketAddress(8888));
ssc.configureBlocking(false);
ByteBuffer buffer = ByteBuffer.allocate(1024);
Selector selector = Selector.open();
ssc.register(selector, SelectionKey.OP_ACCEPT);
while(selector.select() > 0) {
Set<SelectionKey> selectionKeys = selector.selectedKeys();
Iterator<SelectionKey> iterator = selectionKeys.iterator();
while(iterator.hasNext()){
SelectionKey key = iterator.next();
if(key.isAcceptable()) {
//注册关心的事件
SocketChannel sc = ssc.accept();
sc.configureBlocking(false);
sc.register(selector, SelectionKey.OP_READ);
} else if (key.isReadable()) {
SocketChannel channel = (SocketChannel) key.channel();
int len = 0;
while((len = channel.read(buffer)) > 0) {
buffer.flip();
System.out.println(new String(buffer.array(), 0, len));
buffer.clear();
}
}
iterator.remove();
}
}
}
}
//CharsetDemo.java
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CharsetEncoder;
public class CharsetDemo {
public static void main(String[] args) throws IOException {
Charset charset = Charset.forName("UTF-8");
CharsetEncoder encoder = charset.newEncoder();
CharBuffer charBuffer = CharBuffer.allocate(1024);
charBuffer.put("醉里挑灯看剑");
charBuffer.flip();
ByteBuffer buffer = encoder.encode(charBuffer);
System.out.println("编码之后的结果:");
for(int i=0;i<buffer.limit();i++){
System.out.print(buffer.get());
}
System.out.println();
buffer.flip();
CharsetDecoder decoder = charset.newDecoder();
System.out.println("解码之后的结果:");
System.out.println(decoder.decode(buffer));
}
}
//PathDemo.java
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
public class PathDemo {
public static void main(String[] args) throws IOException {
Path path = Paths.get("C:\\2022\\01.txt");
System.out.println(path);
path = Paths.get("C:\\2022", "01");
System.out.println(path);
System.out.println(path.normalize());
Path directory = Files.createDirectory(path);
System.out.println(directory);
Files.copy(path, directory);
}
}