首页 > 代码库 > 深入理解JAVA I/O系列四:RandomAccessFile

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

一、简述

 技术分享

这个是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 Person
11 {
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 IOException
26     {
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 Person
18 {
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 IOException
33     {
34         raf.writeInt(id);
35         raf.writeUTF(name);
36         raf.writeDouble(height);
37     }
38     
39     public void read(RandomAccessFile raf) throws IOException
40     {
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 IOException
2     {
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 IOException
14     {
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、这里插入内容的原理就是:先把插入点后面的内容读入缓冲区,等插入完成后,再讲缓冲区的内容追加到文件的后面。

 

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