代码改变世界

Java中的文件读写

2012-10-28 00:18  chloe_zhou  阅读(3706)  评论(0编辑  收藏  举报

 序言

  之前做项目的时候经常使用文件读写相关的功能,但当时忙于赶进度,只是慌忙地copy实现功能,并没有仔细分析其运作原理,今天来仔细学习一下。

 正文

1.文件读写相关的类

(1) java.io.InputStream: 抽象类,输入字节流,可以将本输入流中的字节读取出来。

  主要方法: void read(byte[] b):从输入流中读取一定数量的字节,将其存储在缓冲区数组b中。

 

(2) java.io.OutputStream: 抽象类,输出字节流,可以接收待输出的字节并将这些字节发送到某个接收器。

     主要方法: void write(byte[] b):接收byte数组中的字节,将byte数组中的字节写入此输出流。

 

(3) FileInputStream: 是InputStream的子类,文件输入流,可以从文件系统中的某个文件中获得输入字节。

  主要方法: FileInputStream FileInputStream(File file):通过打开一个到实际文件的连接来创建一个文件输入流,该文件通过File对象指定。

          void read(byte[] b):将此文件输入流中将最多b.length个字节的数据读入一个byte数组b中。

 

(4) FileOutputStream: 是OutputStream的子类,文件输出流,可以从byte数组中接收数据并将数据写入某个文件。

  主要方法: FileOutputStream FileOutputStream(File file):创建一个准备向file所表示的文件中写入数据的文件输出流。

       void write(byte[] b):将byte数组b中的字节写入此文件输出流。

 

(5) File: 文件或目录路径名的抽象表示形式,此类的实例可能表示实际的文件系统对象(一个文件或一个目录),也可能不表示实际的文件系统对象。File类的实例时不可变的,一旦创建,其对象表示的抽象路径名将永不改变。

  主要方法: File File(String pathname):通过将给定的路径名字符串pathname转换为抽象路径来创建一个新的File实例。

        boolean delete():删除此抽象路径名表示的文件或目录。

        boolean exists():测试此抽象路径名表示的文件或目录是否实际存在。

        String getAbsolutePath():返回此抽象路径名的绝对路径名字符串(带盘符)。

        String getPath():将此抽象路径名转换成一个路径名字符串。

        String getParent():返回此抽象路径名的父目录的路径名字符串,如果没有父目录则返回null。

        boolean isDirectory():测试此抽象路径名表示的文件是否是一个目录,只有是目录且实际存在才会返回true。

        boolean isFile():测试此抽象路径名表示的文件是否是一个标准文件,只有是标准文件且实际存在才会返回true。

        String[] list():返回一个字符串数组,这些字符串指定此抽象路径名所表示的目录中的文件了目录,即返回一个文件夹下所有的文件名和文件夹名。

        boolean mkdir():创建此抽象路径名指定的目录,即创建文件夹。如果此路径中有不存在的父文件夹,则创建失败,返回false。

             boolean mkdirs():创建此抽象路径名指定的目录,包括所有必需但不存在的父目录。

        boolean createNewFile():当且仅当不存在此抽象路径名指定的文件时,不可分地创建一个新的空文件。

 

(6) DataInputStream: java.io.FilterInputStream的子类,数据输入流,允许应用程序以与机器无关的方式从底层输入流中读取基本Java数据类型。

  主要方法: DataInputStream DataInputStream(InputStream in):使用指定的底层InputStream创建一个DataInputStream。

       String readUTF():读入一个使用UTF-8格式编码的字符串,以String形式返回此字符串。

 

(7) DataOutputStream: java.io.FilterOutputStream的子类,数据输出流,允许应用程序以适当的方式将Java基本数据类型的数据写入输出流中。

  主要方法: DataOutputStream DataOutputStream(OutputStream out):使用指定的底层OutputStream创建一个新的数据输出流。

        void writeUTF(String str):以与机器无关的方式使用UTF-8编码将一个字符串写入基础输出流。

 

(8) InputStreamReader: java.io.Reader的子类,是字节流(文件)通向字符流的桥梁,可以使用指定的字符集读取字节,并将其解码为字符。

  主要方法: InputStreamReader InputStreamReader(InputStream in,CharSet c s):由输入流对象创建使用指定字符集的InputStreamReader。

        int read():读取单个字符。

 

(9) OutputStreamWriter: java.io.Writer的子类,是字符流通向字节流(文件)的的桥梁,可使用指定的字符集将要写入流中的字符编码成字节。

  主要方法: OutputStreamWriter OutputStreamWriter(OutputStream out,Charset cs):由输出流对象创建使用指定符集的OutputStreamWriter。

        void write(int c):写入单个字符。

        void write(String str,int off,int len):写入字符串的一部分到文件中。

 

(10) BufferedReader: java.io.Reader的子类,从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。

   主要方法: BufferedReader BufferedReader(Reader in):创建一个使用默认大小输入缓冲区的缓冲字符输入流。

         int read():读取单个字符。

        String readLine():读取一行文本。

 

(11) BufferedWriter: java.io.Writer的子类,将文本写入字符输出流,缓冲各个字符,提供单个字符、数组和字符串的高效写入。

   主要方法: BufferedWriter BufferedWriter(Writer out):创建一个使用默认大小输出缓冲区的缓冲字符输出流。

        void write(int c):写入单个字符。

        void write(String s,int off,int len):写入字符串的某一部分到文件中。

        void newLine():写入一个行分隔符。

 

(12) FileReader: java.io.InputStreamReader的子类,是用来读取字符文件的便捷类,用于读取字符流。此类的构造方法默认的字符编码和默认字节缓冲区大小都是适当的。

    主要方法: FileReader FileReader(File file):为给定的File对象创建一个新的FileReader。

 

(13) FileWriter: java.io.OutputStreamWriter的子类,是用来写入字符文件的便捷类,用于写入字节流。如果要写入原始字节流,则考虑使用FileOutputStream。

       主要方法: FileWriter FileWriter(File file):为给定的FIle对象创建一个新的FileWriter。

 

(14) PrintWriter: java.io.Writer的子类,向文本输出流打印对象的格式化表示形式,不包含用于写入原始字节的方法。

   主要方法: PrintWriter PrintWriter(File file,String csn): 使用指定文件和字符集创建一个不具有自动行刷新的新PrintWriter对象。

                    PrintWriter PrintWriter(OutputStream out,boolean autoFlush):根据现有的OutputStream创建新的PrintWriter对象。

                    PrintWriter PrintWriter(Writer out,boolean autoFlush):根据Writer创建新的PrintWriter。

                    PrintWriter append(CharSequence):将指定的字符序列添加到此writer。

       void println(String x):打印字符串到文件中,然后终止该行。

         void write(String s):写入字符串道文件中。

 

(15) PrintStream: java.io.FilterOutputStream的子类,为其他的输出流添加了功能,使他们能够方便地打印各种数据值表示形式。与其他输出流不同,PrintStream永远不会抛出IOException。 

   主要方法:PrintStream PrintStream(File file,String csn):创建具有指定名称和字符集且不带自动行刷新的打印流。

              PrintStream PrintStream(OutputStream out,boolean autoFlush):根据已有的OutputStream对象创建新的打印流。

         PrintStream append(CharSequence csq):将指定字符序列添加到此输出流。

         void println(String x):打印一行字符串。

         void write(byte[] buf,int off,int len):将len字节从知道那个的初始偏移量为off的byte数组中写入此流。     

 

2. 读取文件byte内容并输出到控制台

 

  读取文件byte内容的处理流程:由文件的File对象生成FileInputStream对象 --> FileInputStream对象调用read函数将文件内容读入byte数组 --> 输出byte数组中的内容到控制台。其实现源代码如下所示,

 1 import java.io.FileInputStream;
 2 import java.io.IOException;
 3 import java.util.Arrays;
 4 
 5 import org.dom4j.DocumentException;
 6 
 7 public class FileIO 
 8 {
 9     void readFile(String filepath)throws IOException
10     {
11         File f=new File(filepath);
12         if(!f.exists())
13         {
14             System.out.println("文件"+f.getAbsolutePath()+"不存在!");
15             return;
16         }
17         else
18         {
19             FileInputStream fis=new FileInputStream(f);//创建对应f的文件输入流
20             byte[] b=new byte[(int)f.length()];//创建一个长度等于文件f长度的byte数组,用于存放从文件中读出的数据
21             fis.read(b);
22             String str=Arrays.toString(b);
23             System.out.println(str);
24             fis.close();   
25         }
26     }
27     
28     public static void main(String args[])throws IOException, DocumentException
29     {
30         FileIO fio=new FileIO();
31         fio.readFile("WebRoot\\accessed_files\\test.txt");  
33     }
34 }

   其中test.txt中只有一行内容:“你好,我是chloe”,运行程序输出的byte数组内容为

[-60, -29, -70, -61, -93, -84, -50, -46, -54, -57, 99, 104, 108, 111, 101]

3. 向文件中写入byte数据

  向文件中输入byte数据的处理过程:由文件的File对象生成FileOutputStream对象 --> FileOutputStream对象调用write函数将byte数组中的内容写入文件。其实现源代码如下

 1 package chloe.fileio;
 2 import java.io.File;
 3 import java.io.FileInputStream;
 4 import java.io.FileOutputStream;
 5 import java.io.IOException;
 6 import java.util.Arrays;
 7 
 8 import org.dom4j.DocumentException;
 9 
10 public class FileIO 
11 {    
12     void writeToFile(String filepath,byte b[])throws IOException
13     {
14         File f=new File(filepath);
15         if(!f.exists())
16         {
17             f.createNewFile();
18             System.out.println("文件"+f.getPath()+"已创建");
19         }
20         FileOutputStream fos=new FileOutputStream(f);
21         fos.write(b);
22         System.out.println("文件内容写入完毕");
23     }
24     
25     public static void main(String args[])throws IOException, DocumentException
26     {
27         FileIO fio=new FileIO();
28         byte[] b=new byte[15];
29         b[0]=(byte)-60;
30         b[1]=(byte)-29;
31         b[2]=(byte)-70;
32         b[3]=(byte)-61;
33         b[4]=(byte)-93;
34         b[5]=(byte)-84;
35         b[6]=(byte)-50;
36         b[7]=(byte)-46;
37         b[8]=(byte)-54;
38         b[9]=(byte)-57;
39         b[10]=(byte)99;
40         b[11]=(byte)104;
41         b[12]=(byte)108;
42         b[13]=(byte)111;
43         b[14]=(byte)101;
44         fio.writeToFile("WebRoot\\accessed_files\\test1.txt", b);
45                 
46     }
47     
48 }

  运行后打开生成的文件test1.txt,其中的内容为“你好,我是chloe”。

  以上是从文件中读取byte数组数据和向写入byte数组,如果需要直接从文件中读取字符串或者向文件中写入字符串,则要使用类型DataInputStream和DataOutputStream.

4. 使用DataOutputStream和DataInputStream写入和读取字符串

  写入处理流程:由文件的File对象生成FileOutputStream对象 --> 由FileOutputStream对象生成DataOutputStream对象 --> DataOutputStream对象调用writeUTF函数将字符串写入文件

  读取处理流程:由文件的File对象生成FileInputStream对象 --> 由FileInputStream对象生成DataInputStream对象 --> DataInputStream对象调用readUTF函数将文件内容读入字符串 --> 输出字符串内容到控制台

      其实现代码如下所示,

 1     void writeStrToFile(String filepath,String str)throws IOException
 2     {
 3         File f=new File(filepath);
 4         if(!f.exists())
 5         {
 6             f.createNewFile();
 7             System.out.println("文件"+f.getPath()+"已创建");
 8         }
 9         FileOutputStream fos=new FileOutputStream(f);
10         DataOutputStream dos=new DataOutputStream(fos);
11         dos.writeUTF(str);
12         System.out.println("文件内容写入完毕");
13         dos.close();
14         fos.close();
15     }
16     
17     String readStrFromFile(String filepath)throws IOException
18     {
19         String result;
20         File f=new File(filepath);
21         if(!f.exists())
22         {
23             System.out.println("文件"+f.getAbsolutePath()+"不存在!");
24             return "";
25         }
26         else
27         {
28             FileInputStream fis=new FileInputStream(f);//创建对应f的文件输入流
29             DataInputStream dis=new DataInputStream(fis);
30             result=dis.readUTF();//只有先用DataOutputStream的write方法写入合规则的数据后才能正常读出内容
31             dis.close();
32             fis.close();
33             if (result!=null)
34                 return result;
35             else 
36                 return "";
37             
38         }
39     }
40     
41     public static void main(String args[])throws IOException, DocumentException
42     {
43         FileIO fio=new FileIO();
44         fio.writeStrToFile("WebRoot\\accessed_files\\test.txt", "你好,我是Chloe~");
45         System.out.println("读取的文件内容为:"+fio.readStrFromFile("WebRoot\\accessed_files\\test.txt"));
46         
47         
48     }
49     
50 }

  运行后输出如下结果,

1 文件内容写入完毕
2 读取的文件内容为:你好,我是Chloe~

  注意,使用DataInputStream读取文件内容时,只有读取由DataOutputStream写入的数据才会正常,否则会报EOFException,所以我感觉这两个类不是很好用。

5. 使用BufferedReader和BufferedWriter写入和读取字符串

   读取字符串时,只要在创建InputStreamReader时指定正确的编码方式,即其编码方式与文件本身的编码方式一致就可以正确读取文件内容,代码如下,

 1     String readLinesFromFile(String filepath) throws IOException
 2     {
 3         String result="";
 4         String line;
 5         File f=new File(filepath);
 6         if(!f.exists())
 7         {
 8             System.out.println("文件"+f.getAbsolutePath()+"不存在!");
 9             return "";
10         }
11         else
12         {
13             FileInputStream fis=new FileInputStream(f);//创建对应f的文件输入流
14             InputStreamReader isr=new InputStreamReader(fis,"UTF-8");
15             BufferedReader br=new BufferedReader(isr);
16             while((line=br.readLine())!=null)
17             {
18                 result=result+line+"\n";
19             }
20             return result;    
21         }
22     }
23     public static void main(String args[])throws IOException, DocumentException
24     {
25         FileIO fio=new FileIO();
26         System.out.print(fio.readLinesFromFile("WebRoot\\accessed_files\\test.txt"));
27     }

  另外,还可以利用更加便捷的写入字符文件的类FileReader实现上述功能,只用将上面的第13~15行改为:

FileReader fr=new FileReader(f);
BufferedReader br=new BufferedReader(fr);

  向文件中写入字符串时,创建OutputStreamWriter也尽量要指明编码方式,方便之后正确读取。

 1 void writeLinesToFile(String filepath,String[]strs)throws IOException
 2     {
 3         File f=new File(filepath);
 4         if(!f.exists())
 5         {
 6             f.createNewFile();
 7             System.out.println("文件"+f.getPath()+"已创建");
 8         }
 9         FileOutputStream fos=new FileOutputStream(f);
10         OutputStreamWriter osw=new OutputStreamWriter(fos,"UTF-8");
11         BufferedWriter bw=new BufferedWriter(osw);
12         for(int i=0;i<strs.length;i++)
13         {
14             bw.write(strs[i]);
15             bw.newLine();
16         }
17         System.out.println("文件内容写入完毕");
18         bw.close();
19         osw.close();
20         fos.close();
21     }
22     public static void main(String args[])throws IOException, DocumentException
23     {
24         FileIO fio=new FileIO();
25         String[] s={"第一行","第二行","   最后一行"};
26         fio.writeLinesToFile("WebRoot\\accessed_files\\test.txt", s);
27     }

   类似的,可以用简化类FileWriter实现上述功能,只需将上面第9-11行改成:

FileWriter fw=new FileWriter(f);
BufferedWriter bw=new BufferedWriter(fw);