社区编辑申请
注册/登录
Java文件I/O的三种方法
开发 后端
RandomAccessFile是不属于InputStream和OutputStream类系的。实际上,除了实现DataInput和DataOutput接口之外(DataInputStream和DataOutputStream也实现了这两个接口),它和这两个类系毫不相干,甚至都没有用InputStream和OutputStream已经准备好的功能;它是一个完全独立的类,所有方法(绝大多数都只属于它自己)都是从零开始写的。

之前在面试中被问到过两次Java中文件读入输出怎么写,当时只记得一个大概,没有办法很清晰的说出一个条理,今天特地看出总结了一下这方面的内容,想要写出来给大家分享。

首先文件读入输出流常用有三种:FileInputStream/FileOutputStream,FileReader/FileWriter,RandomAccessFile。下面具体列出一些简单的例子参考:

基础篇:

1.

  1. FileRead fr = new FileReader(filename);  
  2. String s;  
  3. while( (s=fr.readLine())!=null){  
  4. ...  
  5.  
  6. fr.close();  
  7. //FileWriter同理,输出时可用write()函数  
  8. //Java I/O中所有的Reader、Writer都是面向字符流的输出输出 

2.

  1. FileInputStream fi =new FileInputStream(filename);  
  2. int in 
  3. while( (in=fi.read())!=-1){  
  4. ...  
  5.  
  6. fi.close();  
  7. //FileOutputStream同理  
  8. //Java I/O中所有的Reader、Writer都是面向字节流的输出输出 

3.

  1. RandomAccessFile ra =new RandomAccessFile(filename,"rw");//后面的参数指定的是  
  2. 打开文件流的方式,“rw”是指读写,“r”是只读,Java不提供只写  
  3. ra.seek(number);//将文件指针移动到number处,这里文件指针可以理解为文件开始读的位置  
  4. ra.skipByte(number);//跳过number个字节  
  5. ra.read();  
  6. ra.close();  
  7. //RandomAccessFile既可以读也可以写,而且可以利用seek()函数指定位置 

下面是百度百科的一些介绍:

RandomAccessFile是不属于InputStream和OutputStream类系的。实际上,除了实现DataInput和DataOutput接口之外(DataInputStream和DataOutputStream也实现了这两个接口),它和这两个类系毫不相干,甚至都没有用InputStream和OutputStream已经准备好的功能;它是一个完全独立的类,所有方法(绝大多数都只属于它自己)都是从零开始写的。这可能是因为RandomAccessFile能在文件里面前后移动,所以它的行为与其它的I/O类有些根本性的不同。总而言之,它是一个直接继承Object的,独立的类。

进阶篇:

在nio中,Java重新实现了I/O流,并且引入了一些新的方法来提高速度。我主要介绍通道、内存映射文件

1.通道:

通道和缓冲器是一个成对的概念,Thinking in Java中的一个例子特别好理解:我们把想要读入的文件看作一个煤矿,数据就是我们想要的煤炭。通道好比是传送煤矿的传送带,我们没有办法直接从传送带上拿走煤炭,只好利用卡车来装载这些煤炭,卡车就是缓冲器,它主要负责从通道中取出数据,传给我们写的程序。***能与通道交互的缓冲器是ByteBuffer。可以看出和通道支持的解析流的方式是字节流。所以它配套使用的是FileInputStream/FileOutputStream,RandomAccessFile

例子:

a.

  1. FileChannel fc =new FileOutputStream(filename).getChannel();  
  2. fc.write(ByteBuffer.wrap("something test".getBytes() ));//这里使用ByteBuffer比较简单,其实ByteBuffer可以利用个put()函数写入byte数组  
  3. fc.close(); 

b.

  1. fc= new FileOutputStream(filename).getChannel();  
  2. ByteBuffer buff = ByteBuffer.allocate(size);//没错,ByteBuffer是不提供显示构造函数的,想要新建一个对象必须利用allocate()函数来分配空间。  
  3. fc.read(buff);  
  4. fc.close(); 

为什么想到要用通道来做I/O呢?主要考虑的是性能问题,通道加缓冲器能够让程序一些读写一定量的字符,而只使用InputStream/OutputStream,Reader/Writer只能一次读写一个字节/字符。而程序在进行I/O时要交给操作系统去解决这部分功能(调用系统调用),减少交给操作系统的次数可以有效的消减I/O花费的时间

2.内存映射文件:

内存映射文件主要的意思其实假定将文件都放入内存中,把它当作非常大的数组来访问,效率特别好。为什么比较好呢?这要从Java虚拟机和操作系统开始说起le(其实我也不太懂,刚才看了一篇文章讲的比较清晰,链接是http://www.360doc.com/content...)这篇文章主要介绍了Java I/O的原理以及内存映射文件的原理。我尝试概括一下:Java I/O主要的实现手段肯定是利用系统调用,而系统调用先将想要使用的文件从硬盘调入到内核的I/O缓冲区中,这次会导入比Java程序想要的文件更多的内容(拷入更多的内容是因为程序的局部性原理,能够得到更好的效率),然后再从内核的I/O缓冲区导入到Java进程自己的私有内存空间中。而内存映射文件放弃了两次拷贝的方法,直接将Java进程的虚拟空间与文件对象构成一个映射,当私有内存空间中找不打想要的内容时发生缺页异常,然后利用更底层的系统调用解决这个问题(其实在I/O的系统调用中也涉及到了缺页异常处理),好处就是减少了一次从内核I/O缓冲区到进程私有地址的开销。

例子:

  1. FileChannel fc = new RandomAccessFile(filename,"rw").getChannel();  
  2. MappedByteBuffer mb = fc.map(FileChannel.MapMode.READ_WRITE,start,length);  
  3. mb.put((byte)'x');  
  4. mb.get();  
  5. fc.close(); 

写到这里突然想到在《孔乙己》中“茴”的4中写法,现在Java打开文件也有了至少五种方法了,每一种都一各有利弊。以后也可以穿着长袍问别人你知道Java读写文件的5种方法么

责任编辑:武晓燕 来源: segmentfault
相关推荐

2013-05-28 10:08:41

IO输出

2019-11-23 17:27:54

IO开源

2015-08-10 14:39:46

2011-09-19 16:17:02

Java

2011-12-19 14:05:01

JavaIO

2013-09-17 15:13:28

IO

2013-09-17 13:43:51

IO

2011-11-17 15:30:34

JavaNIOIO

2011-12-14 15:57:13

2020-10-23 07:56:04

Java中的IO流

2021-03-08 07:23:33

Java IO 系统

2011-12-12 10:33:47

同话题下的热门内容

超全!Python图形界面框架PyQt5使用指南!太强了!Python 开发桌面小工具,让代码替我们干重复的工作!Python居然被用来开发游戏了?盘点你想不到的Python开发场景浅谈Python+requests+pytest接口自动化测试框架的搭建跟着 Guava 学 Java 之 不可变集合Python轻量级Web框架:Bottle库!用 Taichi 加速 Python:提速 100+ 倍!实战 | 如何用 Python 自动化监控文件夹完成服务部署!

编辑推荐

使用Kotlin做开发一个月后的感想面试官问你什么是消息队列?把这篇甩给他!五大自动化测试的Python框架图文详解两种算法:深度优先遍历(DFS)和广度优先遍历(BFS)2018年最流行的十大编程语言,其中包括你用的语言吗?
我收藏的内容
点赞
收藏

51CTO技术栈公众号