深入理解JAVA I/O系列四:RandomAccessFile

来源:互联网 发布:js object to list 编辑:程序博客网 时间:2024/05/17 12:46

一、简述

 

这个是JDK上的截图,我们可以看到它的父类是Object,没有继承字节流、字符流家族中任何一个类。并且它实现了DataInput、DataOutput这两个接口,也就意味着这个类既可以读也可以写。

二、存在的意义

1、是JAVA I/O流体系中功能最丰富的文件内容访问类,它提供了众多方法来访问文件内容。

2、由于可以自由访问文件的任意位置,所以如果需要访问文件的部分内容,RandomAccessFile将是更好的选择。

3、可以用来访问保存数据记录的文件,文件的记录的大小不必相同,但是其大小和位置必须是可知的。

这个类在很多资料上翻译成中文都是:随机访问文件,在中文里,随机是具有不确定的含义,指一会访问这里,一会访问那里的意思。如果以这种语义来解释的话,就会感到很困惑。其实,Random在英文中不仅仅有随机,还有任意的意思。如果中文名为任意访问文件是不是就会更好的理解。任意表示我们可以指定文件中任何一个位置去操作一个文件。

三、DEMO演示

(1)、写入文件

复制代码
 1 public class RandomAccessFileTest 2 { 3     public static void main(String[] args) throws IOException 4     { 5         RandomAccessFile raf = new RandomAccessFile("d:/data.txt","rw"); 6         Person p = new Person(1001,"xiaoming",1.80d); 7         p.write(raf); 8     } 9 }10 class Person11 {12     int id;13     String name;14     double height;15     public Person()16     {17     }18     public Person(int id, String name, double height)19     {20         this.id = id;21         this.name = name;22         this.height = height;23     }24     25     public void write(RandomAccessFile raf) throws IOException26     {27         raf.write(id);28         raf.writeUTF(name);29         raf.writeDouble(height);30     }31 } 
复制代码

执行结果:

 

1、执行结果乱码是由于写入的是二进制文件,这个待会我们再使用程序读取。(这个跟前面介绍的DataInputStream、DataOutputStream类似)

2、第五行中,RandomAccessFile的构造函数除了指定了要写入了文件,还有另外一个参数:mod,主要用来指定打开文件的访问模式。

      

3、读取的方式就是读取基本数据类型,其中第28行使用的方法是:writeUTF(String str)

 使用 modified UTF-8 编码以与机器无关的方式将一个字符串写入该文件,这个方法就是将字符串写入文件,而且不用担心会出现乱码,因为使用的编码方式是UTF-8

 (2)、文件读取

由于刚才写入的是二进制文件,现在使用程序去读取文件:

 

复制代码
 1 public class RandomAccessFileTest 2 { 3     public static void main(String[] args) throws IOException 4     { 5         RandomAccessFile raf = new RandomAccessFile("d:/data.txt", "rw"); 6         Person p = new Person(1001, "xiaoming", 1.80d); 7         p.write(raf);// 写入文件后,任意访问文件的指针在文件的结尾 8  9         raf.seek(0);// 读取时,将指针重置到文件的开始位置。10         Person p2 = new Person();11         p2.read(raf);12         System.out.println("id=" + p2.getId() + ";name=" + p2.getName()13                 + ";height=" + p2.getHeight());14 15     }16 }17 class Person18 {19     int id;20     String name;21     double height;22     public Person()23     {24     }25     public Person(int id, String name, double height)26     {27         this.id = id;28         this.name = name;29         this.height = height;30     }31     32     public void write(RandomAccessFile raf) throws IOException33     {34         raf.writeInt(id);35         raf.writeUTF(name);36         raf.writeDouble(height);37     }38     39     public void read(RandomAccessFile raf) throws IOException40     {41         this.id = raf.readInt();42         this.name = raf.readUTF();43         this.height = raf.readDouble();44     }45     public int getId()46     {47         return id;48     }49     public void setId(int id)50     {51         this.id = id;52     }53     public String getName()54     {55         return name;56     }57     public void setName(String name)58     {59         this.name = name;60     }61     public double getHeight()62     {63         return height;64     }65     public void setHeight(double height)66     {67         this.height = height;68     }69     70 } 
复制代码

执行结果:

id=1001;name=xiaoming;height=1.8

1、在39-43行代码中,由于是按基本数据类型写入和读取,所以在读取的时候一定严格按照写入的顺序。

2、第9行的位置上,由于在写入的时候,导致访问的指针的位置在文件的结尾处,现在读取的时候,需要将访问指针的位置重置到文件开头处。

 (3)、追加内容

复制代码
1 public static void main(String[] args) throws IOException2     {3         RandomAccessFile raf = new RandomAccessFile("D:/out.txt","rw");4         raf.seek(raf.length());5         raf.write("\r\n中国移动阅读基地".getBytes());6     }
复制代码

执行结果:

1、这段程序演示了在文件原有内容的基础上去追加内容。其中seek方法就是将访问指针移动到文件内容的末尾。

2、RandomAccessFile依然只能追加,不能像文件的指定位置插入内容。如果强制将文件记录指针移动到中间位置后开始输出内容,则新的内容会覆盖文件中原有的内容。

3、如果需要向文件指定的位置插入内容,程序需要先把插入点后面的内容读入缓冲区,等插入完成后,再讲缓冲区的内容追加到文件的后面。

(4)、指定位置插入

复制代码
 1     public static void main(String[] args) 2     { 3         try 4         { 5             insert("d:/out.txt",5,"插入的内容"); 6         } 7         catch (IOException e) 8         { 9             e.printStackTrace();10         }11     }12     13     private static void insert(String fileName,long pos,String content) throws IOException14     {15         //创建临时空文件16         File tempFile = File.createTempFile("temp",null);17         //在虚拟机终止时,请求删除此抽象路径名表示的文件或目录18         tempFile.deleteOnExit();19         FileOutputStream fos = new FileOutputStream(tempFile);20         21         RandomAccessFile raf = new RandomAccessFile(fileName,"rw");22         raf.seek(pos);23         byte[] buffer = new byte[4];24         int num = 0;25         while(-1 != (num = raf.read(buffer)))26         {27             fos.write(buffer,0,num);28         }29         raf.seek(pos);30         raf.write(content.getBytes());31         FileInputStream fis = new FileInputStream(tempFile);32         while(-1 != (num = fis.read(buffer)))33         {34             raf.write(buffer,0,num);35         }36     }
复制代码

执行结果:

1、这里插入内容的原理就是:先把插入点后面的内容读入缓冲区,等插入完成后,再讲缓冲区的内容追加到文件的后面。