在Java 7,AsynchronousFileChannel
被添加到了Java NIO中。使用AsynchronousFileChannel
可以实现异步地读取和写入文件数据。
和NIO包中的许多类一样,我们用一个静态open()
方法来打开一个AsynchronousFileChannel。
Path path = Paths.get("data/test.xml"); AsynchronousFileChannel fileChannel = AsynchronousFileChannel.open(path, StandardOpenOption.READ);
第一个参数是一个Path,指向与AsynchronousFileChannel相关联的文件。
第二个参数是一个或多个操作选项,他决定了AsynchronousFileChannel对目标文件做怎样的操作。示例代码中我们使用AsynchronousFileChannel.READ
,表明我们要进行读操作。
AsynchronousFileChannel使用read()
方法来读取数据,但有两种方法。
第一种方式是调用AsynchronousFileChannel的read()方法,返回一个Future类型的对象
Future<Integer> operation = fileChannel.read(buffer, 0)
第一个参数使用了一个ByteBuffer类型的变量,从AsynchronousFileChannel中读入的数据写进ByteBuffer。
第二个参数指示了从文件字节序的什么位置开始读取。
无论读操作是否完成,read()
方法立刻返回,后续可以用Future
的isDone()
方法来检查读操作是否完成。
下面是一个使用示例
// open一个AsynchronousFileChannel AsynchronousFileChannel fileChannel = AsynchronousFileChannel.open(path, StandardOpenOption.READ); // 创建buffer和position ByteBuffer buffer = ByteBuffer.allocate(1024); long position = 0; // 调用read方法,返回Future对象 Future<Integer> operation = fileChannel.read(buffer, position); // 循环,直到read完成为止 while(!operation.isDone()); // 读数据到一个字节数组data,并打印 buffer.flip(); byte[] data = new byte[buffer.limit()]; buffer.get(data); System.out.println(new String(data)); buffer.clear();
使用方法在上面的示例中写的很清楚了。
第二种方法是在read方法参数中使用CompletionHandler读取数据
下面是一个使用示例
fileChannel.read(buffer, position, buffer, new CompletionHandler<Integer, ByteBuffer>() { @Override public void completed(Integer result, ByteBuffer attachment) { System.out.println("result = " + result); attachment.flip(); byte[] data = new byte[attachment.limit()]; attachment.get(data); System.out.println(new String(data)); attachment.clear(); } @Override public void failed(Throwable exc, ByteBuffer attachment) { } });
如上,我们将一个CompletionHandler作为read方法的参数,并重写了completed
方法。当read
操作完成后,这个completed
方法会被调用。
result
参数会是read中读取的字节数。attachment
参数是一个ByteBuffer,存了读取的数据。它其实就是read方法的第三个参数,这里我们使用了ByteBuffer类型,其实别的啥也可以,反正completed()
是自己写。读取失败时,failed()
方法会被调用。
就像读取一样,我们同样有两种方式向 AsynchronousFileChannel 写入数据。我们可以调用它的2个重载的 write() 方法。下面我们将分别加以介绍。
使用Future写入数据
AsynchronousFileChannel也可以异步写入数据,下面是一个完整的示例。
Path path = Paths.get("data/test-write.txt"); AsynchronousFileChannel fileChannel = AsynchronousFileChannel.open(path, StandardOpenOption.WRITE); ByteBuffer buffer = ByteBuffer.allocate(1024); long position = 0; buffer.put("test data".getBytes()); buffer.flip(); // 返回一个Future Future<Integer> operation = fileChannel.write(buffer, position); buffer.clear(); while(!operation.isDone()); System.out.println("Write done");
代码很好看懂,这里就不做解释了。
注意写入的目标文件需要提前准备好,否则会抛出java.nio.file.NoSuchFileException
,我们可以用下面的代码解决这个问题。
if(!File.exists(path)){ Files.createFile(path); }
使用CompletionHandler写入数据
我们也可以使用 CompletionHandler代替Future向AsynchronousFileChannel写入数据,这种方式可以更加直接的知道写入过程是否完成。下面是示例程序
Path path = Paths.get("data/test-write.txt"); if(!Files.exists(path)){ Files.createFile(path); } AsynchronousFileChannel fileChannel = AsynchronousFileChannel.open(path, StandardOpenOption.WRITE); ByteBuffer buffer = ByteBuffer.allocate(1024); long position = 0; buffer.put("test data".getBytes()); buffer.flip(); fileChannel.write(buffer, position, buffer, new CompletionHandler<Integer, ByteBuffer>() { @Override public void completed(Integer result, ByteBuffer attachment) { System.out.println("bytes written: " + result); } @Override public void failed(Throwable exc, ByteBuffer attachment) { System.out.println("Write failed"); exc.printStackTrace(); } });
当写入程序完成时,CompletionHandler的completed()方法将会被调用,相反的如果写入失败则会调用failed()方法。
要留意CompletionHandler的方法的参数 attachemnt是怎么使用的,其实和read中的类似。