偶遇RandomAccessFile
一、前言
本来在研究NIO,别人举的栗子里面,看到一个RandomAccessFile类,之前没见过,就去看了一下,现将相关内容记录如下
二、正文
RandomAccessFile直接继承自Object,并且实现DataInput和DataOutput接口,它不属于InputStream和OutputStream类系的,并不是针对流的操作,它能在读取文件的时候前后移动,这个是和其他针对流操作的I/O类的本质区别。
基本上,RandomAccessFile的工作方式是,把DataInputStream和DataOutputStream粘起来,再加上它自己的一些方法,比如定位用的getFilePointer( ),在文件里移动用的seek( ),以及判断文件大小的length( )。此外,它的构造函数还要一个表示以只读方式("r"),还是以读写方式("rw")打开文件的参数 (和C的fopen( )一模一样),它不支持只写文件。它支持对文件随机访问的读取和写入,即我们可以在指定的位置从文件读取/写入数据。
平时,我们如果要往一个文件末尾追加一些数据,我们会将文件加载到内存,然后进行操作,但是如果文件本身非常大,比如好几G的量,那么普通的电脑承受不了,我们这时就可以使用RandomAccessFile,它无需将文件内容加载在内存,便可以在文件末尾追加内容。当然,有可能很多人会问,这个RandomAccessFile可以在文件指定位置插入数据吗?答案是:RandomAccessFile只能在指定位置修改数据,并不能插入数据(据我所知貌似没有哪个类支持在某个文件某个位置插入数据吧)。
RandomAccessFile在读取等长数据的情况下也有优势。
下面给出RandomAccessFile相关使用实例,先看下要操作的文件内容:
//实例一,从指定位置读取文件 package com.randomaccessfile.demo; import java.io.RandomAccessFile; /** * * @author cyg * @date 2017-6-3 下午02:33:13 * @description 随机读取文件的内容 */ public class Demo1 { public static void main(String[]args) throws Exception{ //文件内容:hello world! RandomAccessFile raf = new RandomAccessFile("C:\\Users\\jackalc\\Desktop\\demo.txt", "r"); System.out.println("在未指定位置之前,文件指针:"+raf.getFilePointer()); raf.seek(6);//指定文件指针位置,从0开始 System.out.println("在指定位置之后,文件指针:"+raf.getFilePointer()); int readLength = 0; byte [] bytes = new byte[1024]; while((readLength=raf.read(bytes))>0){ System.out.println(new String(bytes,0,readLength)); } } } //输出 在未指定位置之前,文件指针:0 在指定位置之后,文件指针:6 world!
//实例二,文件末尾追加内容 package com.randomaccessfile.demo; import java.io.RandomAccessFile; /** * * @author cyg * @date 2017-6-3 下午02:33:13 * @description 文件末尾追加内容 */ public class Demo2 { public static void main(String[]args) throws Exception{ //文件内容:hello world! RandomAccessFile raf = new RandomAccessFile("C:\\Users\\jackalc\\Desktop\\demo.txt", "rw");//读写方式打开 raf.seek(raf.length());//将文件的指针移动到原先内容最后一个字节的后面 raf.write("最末尾追加的内容".getBytes()); } }
实例二运行后,文件内容变更为:
//实例三,在文件指定位置修改文件 package com.randomaccessfile.demo; import java.io.RandomAccessFile; /** * * @author cyg * @date 2017-6-3 下午02:33:13 * @description 在文件指定位置修改文件 */ public class Demo3 { public static void main(String[]args) throws Exception{ //文件内容:hello world! RandomAccessFile raf = new RandomAccessFile("C:\\Users\\jackalc\\Desktop\\demo.txt", "rw");//读写方式打开 raf.seek(6);//"6"是“w”所在的位置 raf.write("我现在在指定的位置修改内容,我会覆盖从指定位置开始,和我等长的文件内容!".getBytes()); } }
实例三运行之后,文件内容变更成这样:
如果想要在文件指定位置插入数据,而不是修改数据,要怎么操作呢?RandomAccessFile并没有提供这个方法,对于这个需求,我们可以将指定位置后面的内容先存储到临时文件里面,然后等内容插入完,我们再将临时文件的内容插入到刚刚插入的内容后面,贴一下别人的代码(原文见“三、连接”处):
//实例四,在文件指定位置插入内容 package com.randomaccessfile.demo; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.RandomAccessFile; /** * * @author cyg * @date 2017-6-3 下午02:33:13 * @description 在文件指定位置插入内容 */ public class Demo4 { public static void main(String[] args) { //文件内容:hello world! String path="C:\\Users\\jackalc\\Desktop\\demo.txt"; insert(path, 6, "这是我需要插入的内容,不覆盖原先的内容"); } /** * 实现向指定位置 * 插入数据 * @param fileName 文件名 * @param points 指针位置 * @param insertContent 插入内容 * **/ public static void insert(String fileName,long points,String insertContent){ try{ File tmp=File.createTempFile("tmp", null); tmp.deleteOnExit();//在JVM退出时删除 RandomAccessFile raf=new RandomAccessFile(fileName, "rw"); //创建一个临时文件夹来保存插入点后的数据 FileOutputStream tmpOut=new FileOutputStream(tmp); FileInputStream tmpIn=new FileInputStream(tmp); raf.seek(points); /**将插入点后的内容读入临时文件夹**/ byte [] buff=new byte[1024]; //用于保存临时读取的字节数 int hasRead=0; //循环读取插入点后的内容 while((hasRead=raf.read(buff))>0){ // 将读取的数据写入临时文件中 tmpOut.write(buff, 0, hasRead); } //插入需要指定添加的数据 raf.seek(points);//返回原来的插入处 //追加需要追加的内容 raf.write(insertContent.getBytes()); //最后追加临时文件中的内容 while((hasRead=tmpIn.read(buff))>0){ raf.write(buff,0,hasRead); } }catch(Exception e){ e.printStackTrace(); } } }
实例四运行完之后,文件内容如下:
三、链接
1、http://blog.csdn.net/czplplp_900725/article/details/37809579
2、http://www.cnblogs.com/skywang12345/p/io_26.html
四、联系本人
为方便没有博客园账号的读者交流,特意建立一个企鹅群(纯公益,非利益相关),读者如果有对博文不明之处,欢迎加群交流:261746360,小杜比亚-博客园