与其他语言相比,Java的IO功能显得异常复杂,各种流操作,通过程序员多次封装才可以达到操作文件的目的。自从jdk1.4之后,java提供了一个新的api完成IO操作,人称New IO(NIO),使用java nio包来进行IO操作,相比原先的API相比要方便多了(每次用老的api写IO操作,心中就会有千万的草泥马飘过)。Jdk1.7发布后,java nio又有了新的改进,人称NIO.2。终于java的IO操作可以和其他语言接轨了T^T。

下面就来了解下java的NIO包(注意了啊~这里我使用的jdk版本是1.7)

探索Path与Files

在java 7 中文件路径和对文件的操作被很清晰地用两个类分开了,路径全部使用Path来处理,而文件的操作则通过Files类来操作。

Path的创建非常简单,通过Paths.get()方法就可以获得了。

Path path=Paths.get("F:\\javawork\\eclipsework");
System.out.println(path.toAbsolutePath());
System.out.println(path.getFileName());
System.out.println(path.getRoot());
System.out.println(path.getParent());
System.out.println(path.getNameCount());

通过这个path对象,就可以获得很多和路径有关的信息了。

为了兼容java 6 ,File类中特别添加了一个toPath() 方法帮助File类获取Ptah对象,于此同时,Path也拥有一个toFile()方法,方便两个类相互转换。

有了Path的帮助,接下来就是处理文件的操作了。我们在获取的了相应的路径之后,现在就需要Files类来操作文件了。

之前提到路径的处理和文件的处理被分的很清楚,那么文件的相关信息,就只有由Files类来完成了。

Path path=Paths.get("F:\\javawork\\eclipsework\\jdk7Test\\src\\pro\\app\\hello2.txt");
System.out.println(Files.getLastModifiedTime(path));
System.out.println(Files.size(path));
System.out.println(Files.isSymbolicLink(path));
System.out.println(Files.isDirectory(path));
System.out.println(Files.readAttributes(path,"*"));

使用Files才完成文件的创建、删除、复制和移动操作也是非常方便。

//path和target均是Path类的对象
Files.createFile(path); //创建文件
Files.delete(path); //删除文件
Files.copy(path, target); //复制文件
Files.move(path, target); //移动文件

同时copy()和move()方法均有第三个参数(通过import static java.nio.file.StandardCopyOption.*;获得常量)。

REPLACE_EXISTING  替换已经存在的文件
COPY_ATTRIBUTES 复制文件属性
ATOMIC_MOVE 确保两边操作均成功,否则回滚

文件的读取与写入

想到要用java对文件操作就很痛苦,有木有~流过来,输出入~泥煤哒!!jdk1.7已经开始解放大家的痛苦了。(当然java原先的IO操作让程序员有更高的自由度,可以非常容易滴处理IO的细节,但有时候,咱真的没有那个需求啊~)。

Path path=Paths.get("test.txt");
List<String> lines=Files.readAllLines(path,StandardCharsets.UTF_8); for(String line:lines){
  System.out.println(line);
}

很简单有木有,如果想通过byte类型获取数据可以这么操作。

byte[] bytes=Files.readAllBytes(path);
for(byte b:bytes){
String temp=new String(new byte[]{b});
System.out.print(temp);
}

读了文件之后,现在就要开始写入文件了,这里使用Files类的write方法~

Path path=Paths.get("F:\\javawork\\eclipsework\\jdk7Test\\src\\pro\\app\\hello2.txt");
String temp="hello world~";
Files.write(path,temp.getBytes(),StandardOpenOption.APPEND);

不要太轻松啊~啊哈哈哈哈哈。

异步IO

NIO给我们带来最大的惊喜就是异步的IO操作了,相比传统的IO方式,NIO通过异步操作极大地提升了IO的通信。异步IO允许程序在结束IO操作之前,处理其他的事情。这样就可以充分地利用CPU了。想象一下在完成一个100G的文件操作时,如果是老版本的api,我们在完成IO操作之前,除了等待还是无奈的等待~~~

新的异步IO主要有两种形式,将来式和回调式(有种学英语的感觉orz)。我们先来看看将来式。

Path path=Paths.get("F:\\javawork\\eclipsework\\jdk7Test\\src\\pro\\app\\hello2.txt");
//异步打开文件
try(AsynchronousFileChannel channel=AsynchronousFileChannel.open(path)){
  ByteBuffer buffer=ByteBuffer.allocate(100_000);
Future<Integer> result=channel.read(buffer, 0);
//使用isDone()判断result是否结束
while(!result.isDone()){
  //do something else
}
buffer.flip();
while(buffer.hasRemaining()){
  System.out.print((char) buffer.get());
}
//clear()方法会清空整个缓冲区
buffer.clear();
Integer byteRead=result.get();
System.out.println(byteRead);
}catch(IOException | ExecutionException | InterruptedException e){
  System.out.println(e.getMessage());
}

在完成读取之前,我们可以在 while(!result.isDone())中完成其他的事儿~

这里可以复习下nio的一些概念,Java NIO 是面向缓存Buffer的IO操作方式。

Buffer的本质是一块可以读写数据的内存,NIO将其封装成一个对象。可以将它视作一个集装箱,咱把货物装好了,打包就可以从通道送出去了。

而Channel是IO操作中的通道,我们通过通道获得数据,输出数据,好比是马路,我们想要搬运货物都得通过这个通道。

前面提到Buffer是一块可以读写的内存,一般我们使用Buffer有这么几个步骤:

1、  声明Buffer大小
2、 写入数据到buffer
3、 调用filp()方法
4、 从Buffer中读取数据,放到相应的位置
5、 调用clear()或者compact()方法清理Buffer的内存空间

Buffer有这么几个类型:ByteBuffer、MappedByteBuffer、CharBuffer、DoubleBuffer、FloatBuffer、IntBuffer、LongBuffer、ShortBuffer

很惊讶有木有!这就是java的基本类型加了个Buffer啊,想要什么类型的数据,咱就用什么类型的Buffer,每次使用之前就通过allocate()方法来给Buffer分配大小。

再有,之前的jdk版本(1.7之前的版本)写IO操作最怕忘记写close()方法了,在新的api中try语句之后可以接上一个括号,在括号中声明需要保护的资源,然后交给java来关闭这个资源。

接下来再看看回调式,比起将来式的写法,这种更加自然一些。有点像ajax的写法。

Path path=Paths.get("test.txt");
try(AsynchronousFileChannel channel=AsynchronousFileChannel.open(path);){
  ByteBuffer buffer=ByteBuffer.allocate(100_000);
channel.read(buffer,0,buffer,new CompletionHandler<Integer,ByteBuffer>(){
  @Override
public void completed(Integer result,ByteBuffer attachment){
  System.out.println(result);
}
@Override
public void failed(Throwable exc, ByteBuffer attachment) {
  System.out.println(exc.getMessage());
}
});
}catch(IOException e){
  System.out.println(e.getMessage());
}

主线程派出一个侦查员CompletionHandler,来观察独立线程的IO操作,如果IO操作成功则调用CompletionHandler中的completed()中的方法,如果失败就调用failed方法,采取行动挽救损失。

总结

Path类与Files类

文件的写入与读取

IO的异步

Please Call Me NIO的更多相关文章

  1. 源码分析netty服务器创建过程vs java nio服务器创建

    1.Java NIO服务端创建 首先,我们通过一个时序图来看下如何创建一个NIO服务端并启动监听,接收多个客户端的连接,进行消息的异步读写. 示例代码(参考文献[2]): import java.io ...

  2. BIO\NIO\AIO记录

    IO操作可以分为3类:同步阻塞(BIO).同步非阻塞(NIO).异步(AIO). 同步阻塞(BIO):在此种方式下,用户线程发起一个IO操作以后,必须等待IO操作的完成,只有当真正完成了IO操作以后, ...

  3. 支撑Java NIO 与 NodeJS的底层技术

    支撑Java NIO 与 NodeJS的底层技术 众所周知在近几个版本的Java中增加了一些对Java NIO.NIO2的支持,与此同时NodeJS技术栈中最为人称道的优势之一就是其高性能IO,那么我 ...

  4. Java I/O and NIO [reproduced]

    Java I/O and NIO.2---Five ways to maximize Java NIO and NIO.2---Build more responsive Java applicati ...

  5. JAVA NIO学习笔记1 - 架构简介

    最近项目中遇到不少NIO相关知识,之前对这块接触得较少,算是我的一个盲区,打算花点时间学习,简单做一点个人学习总结. 简介 NIO(New IO)是JDK1.4以后推出的全新IO API,相比传统IO ...

  6. Java NIO概述

    Java NIO 由以下几个核心部分组成: Channels Buffers Selectors 虽然 Java NIO 中除此之外还有很多类和组件,但在我看来,Channel,Buffer 和 Se ...

  7. JAVA NIO Socket通道

      DatagramChannel和SocketChannel都实现定义读写功能,ServerSocketChannel不实现,只负责监听传入的连接,并建立新的SocketChannel,本身不传输数 ...

  8. JAVA NIO FileChannel 内存映射文件

      文件通道总是阻塞式的. 文件通道不能创建,只能通过(RandomAccessFile.FileInputStream.FileOutputStream)getChannel()获得,具有与File ...

  9. java nio系列文章

    java nio系列教程 基于NIO的Client/Server程序实践 (推荐) java nio与并发编程相关电子书籍   (访问密码 48dd) 理解NIO nio学习记录 图解ByteBuff ...

  10. (转)NIO与AIO,同步/异步,阻塞/非阻塞

    原文地址: http://www.cnblogs.com/enjoy-ourselves/p/3793771.html 1.flip(),compact(),与clear()的使用 flip()内部实 ...

随机推荐

  1. 关于Charles抓取手机访问的Https请求

    准备工作 本次测试的Charles版本为3.9.1 · 首先在Charles中开启HTTP请求的远程监听. · 然后分别在手机和Mac上安装Charles的证书. 注意:证书一定要一致,否则抓取不到. ...

  2. Drools 查询学习

    Drools 查询学习查询以 query 关键字开始,以 end 关键字结束,在 package 当中一个查询要有唯一的名称,查询的内容就是查询的条件部分,条件部分内容的写法与规则的 LHS 部分写法 ...

  3. Qtablevies获取内容

    首先是向tableview中添加内容 model=new QStandardItemModel(); model->setHorizontalHeaderItem(, new QStandard ...

  4. hdu2662

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2662 莫名其妙写了一个题,感觉还是很有价值的记录一下. 题目大意:给两个互质的数,求用无限个它们不能组 ...

  5. js整理5

    proto 每个对象具有的属性,指向构造该对象的构造函数的原型对象 prototype 函数的特有属性,指向原型对象:原型对象可以是对象,数组,函数等类型: constructor 原型对象和实例,都 ...

  6. 商贸食品车销成功应用PDA抄单 现场开单 打印销售单安卓智能手持POS应用

    中小超市配送食品,酒水饮料,业务员以往是挨家挨户抄每个超市需要哪些东西,晚上回公司再统计,打到软件里面,开单配货. 选用PDA后,人手一台,直接在超市里面抄好货物,通过网络传输到公司软件上面,加快了工 ...

  7. C# 解析JSON的几种办法

    欲成为海洋大师,必知晓海中每一滴水的真名. 刚开始只是想找一个转换JSON数组的方法,结果在MSDN翻到一大把. 搜索过程中免不了碰到一大堆名词:WCF => DataContract => ...

  8. Trigger和ViewStateManager的具体比较

    ViewStateManager的好处  拥有 GeneratedDuration ,可以很方便的进行几个状态之间的切换过渡动画. 坏处是,在界面加载时只能显示默认效果,通过GoToStateActi ...

  9. Mysql在windows系统下的配置

    因为项目测试需求,不得不在本地装一个Mysql才能更方便地进行程序调试,整个过程虽然简单,但也遇到了一点麻烦,所以贴出来当是备忘. 这里采用MySQL Community Server  5.7.12 ...

  10. Django--全文检索功能

    经过两个月的时间,毕设终于算是把所有主要功能都完成了,最近这一周为了实现全文检索的功能,也算是查阅了不少资料,今天就在这里记录一下,以免以后再用到时抓瞎了~ 首先介绍一下我使用的Django全文检索逻 ...