java处理流,IO

字符流的抽象基类是:Writer 和 Reader !

用于操作字符流的子类对象有:

1. FileReader  和 FileWriter .

        2. 功能对象:BufferedReader  和 BufferedWriter  他们分别对应于:FileReader 和 FileWriter ,用来达到对FileReader  和 FileWriter 的高效操作,具体见下文讲解。

——————————————————————————————————————————

 

【FileWriter】的学习,将从如下5点展开,并通过具体示例来体现:

1.      基本方法的演示

2.      对象创建的细节。

3.       对象操作的细节:close()  write()  flush()

4.      续写换行

5.      文件处理是的——异常处理规范!

 

示例一:将一个段文字数据写入到硬盘上.

 

[java]   view plain copy print ?
  1. import java.io.FileWriter;  
  2. import java.io.IOException;  
  3.   
  4. public class FileWriterDemo {  
  5.     public static void main(String[] args) throws IOException {  
  6.         /*   
  7.         //需求:将一个段文字数据写入到硬盘上. 
  8.         思路: 
  9.         1,一段文字就是字符串数据。 
  10.         2,写到硬盘上,从哪到哪呢?字符串数据在内存中,写到硬盘上——将内存中的数据搞到硬盘上, 
  11.         这就涉及到了设备之间的数据处理。就要用到IO技术。 
  12.         既然是从内存到硬盘,应该是输出流。 
  13.         3,对于文字而言,io中提供了便捷的操作,比如字符流。 
  14.         4,结合两者,需要输出流,需要字符流,可以使用字符输出流。Writer 
  15.         5,具体用哪个子类对象呢?硬盘上用于存储数据的体现:文件。在这个体系中有对象FileWriter 。  
  16.          */  
  17.           
  18.           
  19.         //1,通过FileWriter创建流对象。构造时,必须明确写入数据需要存储的位置。   
  20.         /* 
  21.          * 该对象一创建,目的文件就会被创建。 
  22.          * 如果该文件已经存在,会被覆盖。  
  23.          * 做了什么事呢?在堆内存中创建一个对象。同时调用了系统的资源。 
  24.          */  
  25.         FileWriter fw = new FileWriter("demo.txt");  
  26.           
  27.         //2,使用字符输出流对象将字符串进行写入。调用写入方法。  
  28.         //数据没有直接写入到目的文件中,而是写入到了临时缓冲中。  
  29.         fw.write("abcdef");  
  30.           
  31.         //3,怎么把数据弄到文件中呢?发现Writer类中有一个flush()方法。刷新缓冲区,将缓冲的数据立即写入到目标中。  
  32.         fw.flush();  
  33.           
  34.         fw.write("haha");         
  35.         //4,关闭此流,关闭资源。在关闭之前,先刷一次缓冲,将数据都写入到目的中。  
  36.         fw.close();  
  37.                   
  38.         /* 
  39.          * flush()和close()有什么区别? 
  40.          * flush():仅将缓冲中的数据刷新到目的地。流对象可以继续使用。可以使用多次。 
  41.          * close():将缓冲中的数据刷到目的地后,直接关闭流资源,流无法继续使用。只能使用一次。 
  42.          * 在close()方法当中其实在关闭之前都会调用一次flush(); 
  43.          *  
  44.          */  
  45.     }  
  46. }  

 

通过上述示例,我们必须注意以下几点:

1.      通过FileWriter创建流对象。构造时,必须明确写入数据需要存储的位置。

2.      使用字符输出流对象将字符串写入时,必须调用 writer() 方法,但是这些字符并没有直接被写到 目的地中,而是被写在了Writer类默认的缓冲区中,这可以在Java源代码中查到。

3.      将缓冲区中的数据写到目的中,需要调用flush()方法,将缓冲区刷新。

4.      在一系列操作完成后,必须要做的动作是,关闭流,释放系统资源。需调用close()。

5.      其次就是,flush()和 close()方法调用,这两者之间的区别:上文已经提到,不在赘述。

 

示例二:想对刚才的文件demo.txt 进行一个续写!

 

[java]   view plain copy print ?
  1. import java.io.FileWriter;  
  2. import java.io.IOException;  
  3.   
  4. public class FileWriterDemo2 {  
  5.   
  6. private static final String LINE_SPARATOR = System.getProperty("line.separator");  
  7.   
  8.     public static void main(String[] args) throws IOException {  
  9.   
  10.         /* 
  11.          * 需求:想要来个续写。 
  12.          * 这个对象创建是不行的,因为该构造一创建,就覆盖了已有的文件。  
  13.          * 可以使用另一个构造函数,加入一个boolean参数,为true,就可以实现续写。 
  14.          *  
  15.          * 需求:想要将数据分行写入。 
  16.          *  
  17.          * window中的特有软件比如notepad。只识别window中的特有换行 \r\n. 
  18.          * 为了在不同系统平台下换行。使用System类获取当前系统信息。 
  19.          *  
  20.          *  
  21.          */  
  22.         //该构造方式,通过第二个参数来决定是否进行续写  
  23.         FileWriter fw = new FileWriter("demo2.txt",true);  
  24.         fw.write("xi"+LINE_SPARATOR+"xi");  
  25.         fw.close();               
  26.     }  
  27. }  

 

上述示例主要提示我们一下几点:

1.    如果想在原来的文本后,在继续添加文字,该怎么办?这时,可以使用FileWriter的另外一个构造方法,传入加入一个boolean参数,为true,就可以实现续写。

2.    其次,就是,当数据需要分行写入时,为了让程序在不同的系统平台上都可以运行,不能讲换行的符号写死,可以动态的获取系统的属性来取到当前系统中的换行符:

private static final String LINE_SPARATOR =      System.getProperty("line.separator");

3.    无论如何都不能忘记:流不再使用的时候,必须得关闭。

 

 

在对IO系列中的FileWriter对象有了大概的了解后,我们会发现,在程序中的很多地方都会发生异常:繁多而又不同的异常。因此,我们有必要来介绍一下,Java中IO体系的异常处理规范:当然了,也是从一个示例讲起:

 

[java]   view plain copy print ?
  1. import java.io.FileWriter;  
  2. import java.io.IOException;  
  3. public class FileWriterDemo3 {  
  4.       
  5.     public static void main(String[] args) {  
  6.         /* 
  7.          * IO异常的处理规范。  
  8.          * 创建流对象—————— 在try外创建流对象的引用。 在try内对流对象进行初始化。 
  9.          */  
  10.         FileWriter fw = null;  
  11.         try {  
  12.             fw = new FileWriter("k:\\demo3.txt");  
  13.   
  14.             fw.write("abcde");  
  15.             fw.flush();  
  16.   
  17.         } catch (IOException e) {  
  18.   
  19.             System.out.println(e.toString());  
  20.         } finally {  
  21.             //在关闭之间必须得判断这个流是否存在,是否被创建。  
  22.             if (fw != null)  
  23.                 try {  
  24.                     fw.close();  
  25.                 } catch (IOException e) {  
  26.   
  27.                     // 相关的代码处理。比如说,将关闭失败的信息记录到日志文件中。  
  28.                     throw new RuntimeException("关闭失败");  
  29.                 }  
  30.         }  
  31.     }  
  32. }  

 

通过以上示例,我们的明白,在Java中IO异常的处理时:

1.    对象的创建方式——创建流对象时, 在try外创建流对象的引用。 在try内对流对象进行初始化。

2.    在finally的代码中,流的关闭时必须得执行的动作。但是,在关闭前,必须得判断,这个流是否创建成功,如果没有创建成功的话,何谈关闭。

3.    在处理流关闭失败后,可以进行一系列的操作:或者将关闭失败的信息记录到日志文件中。


 

————————————————————————————

 

下面我们来讲解:另一个重要的对象——FileReader。

 FileReader: 这个对象的主要用途在于读取,因此,我们将着重分析,它对指定文件的读取方式,FileReader 读取字符时有两种方式:

1. 读取单个字符的方法:读取单个字符。在字符可用、发生 I/O 错误或者已到达流的末尾前,此方法一直阻塞。返回值:作为整数读取的字符,范围在 0 到 65535 之间 (0x00-0xffff),如果已到达流的末尾,则返回 -1

2. 读取字符数组:将读到的字符存储到字符数组里面,并且返回字符数组的长度。如果到达流的末尾,返回-1.

下面来演示,他们的具体应用:

 

[java]   view plain copy print ?
  1. import java.io.FileNotFoundException;  
  2. import java.io.FileReader;  
  3. import java.io.IOException;  
  4.   
  5. public class FileReaderTest {  
  6.   
  7.     public static void main(String[] args) throws IOException {  
  8.         // TODO Auto-generated method stub  
  9.         FileReader fr = new FileReader("IO流.txt");  
  10.           
  11.         //演示读取单个字符  
  12.         long time = System.currentTimeMillis();  
  13.         readChar(fr);  
  14.         long timereadChar = System.currentTimeMillis();  
  15.           
  16.         System.out.println("time Read char is  = " + (timereadChar-time));  
  17.         //演示读取字符到缓冲区中。  
  18.           
  19.         long time2 = System.currentTimeMillis();  
  20.         readToBuf(fr);  
  21.         long timeReadBuf = System.currentTimeMillis();  
  22.         System.out.println("time Read to Buf is =  " + (timeReadBuf-time2));  
  23.           
  24.   
  25.     }  
  26.     private static void readToBuf(FileReader fr) throws IOException {  
  27.         //定义一个字符缓冲区,用于存放读到的字符。  
  28.         char[] buf = new char[50];  
  29.         //设刚开始读到的字符为0  
  30.         int len = 0 ;  
  31.         //一直循环读取字符到缓冲区中,直到读到流的末尾。  
  32.         while((len = fr.read(buf)) != -1){  
  33.             //将每次读满的缓冲区中的字符,变成字符串打印出来。  
  34.             System.out.println(new String(buf , 0 , len));        
  35.         }     
  36.     }  
  37.     private static void readChar(FileReader fr) throws IOException {  
  38.         //设每个读取到的字符整数值为ch.  
  39.         int ch = 0;   
  40.         //循环读取字符,直到流的末尾  
  41.         while((ch = fr.read()) != -1){  
  42.             //将读取到的字符,强制转换为 char  
  43.             System.out.print((char) ch);  
  44.         }  
  45.     }  
  46. }  

 

上述两种读取方式,最明前的区别在于:

    第二种读取字符的方式,要比第一种方式高效。

 

 

到目前为止,我们已经学了FileReader, FileWrite的具体使用,以及IO异常的处理方式,下面将通过一个具体的示例,来综合的应用上述所学的知识:

我们将完成一个:文件复制的练习。并且使用两种读取字符方式来完成。该例子中用的路径,假设在你的机器中也存在。

 

 

题目:

   

        * 练习:将c盘的一个文本文件复制到d盘。

        * 思路:

        * 1,c盘的文件是一个数据的源。

        * 复制到的d盘,说明d盘是数据的存储的目的。该目的中应该有一个文件。

        * 2,先读取c盘的文件。将读取到的数据写入到目的地中。

        *3,既然是操作文本文件,只要使用字符流即可。

 

 

1.用缓冲区数组,来完成读写操作;

[cpp]   view plain copy print ?
  1. import java.io.FileReader;  
  2. import java.io.FileWriter;  
  3. import java.io.IOException;  
  4.   
  5. public class CopyTextTest1 {  
  6.   
  7.       
  8.     public static void main(String[] args) {  
  9.   
  10.         /* 
  11.          *  
  12.          * 使用缓冲区数组。 
  13.          * 使用的就是可以操作数组的读写方法。 
  14.          */  
  15.         //1,定义字符输入流和字符输出流的引用。   
  16.         FileReader fr = null;  
  17.         FileWriter fw = null;  
  18.           
  19.         try {  
  20.             //2,对流对象进行初始化。  
  21.             fr = new FileReader("demo.txt");  
  22.             fw = new FileWriter("copy_demo2.txt");  
  23.               
  24.             //3,定义一个数组缓冲区。用于缓冲读取到的数据。   
  25.             char[] buf = new char[1024];  
  26.               
  27.             //4,读写操作。   
  28.             int len = 0;  
  29.             while((len = fr.read(buf))!=-1){  
  30.                 fw.write(buf,0,len);  
  31.             }  
  32.               
  33.         } catch (Exception e) {  
  34.             System.out.println(e.toString());  
  35.         }finally{  
  36.               
  37.             if(fw!=null)  
  38.                 try {  
  39.                     fw.close();  
  40.                 } catch (IOException e) {  
  41.                       
  42.                     throw new RuntimeException("写入关闭失败");  
  43.                 }  
  44.               
  45.             if(fr!=null)  
  46.                 try {  
  47.                     fr.close();  
  48.                 } catch (IOException e) {  
  49.                       
  50.                     e.printStackTrace();  
  51.                 }  
  52.         }  
  53.     }  
  54. }  

2. 用读取单个字符的方式来完成。

[java]   view plain copy print ?
  1. import java.io.FileReader;  
  2. import java.io.FileWriter;  
  3. import java.io.IOException;  
  4.   
  5. public class CopyTextTest {  
  6.   
  7.     /** 
  8.      * @param args 
  9.      * @throws IOException  
  10.      */  
  11.     public static void main(String[] args) throws IOException {  
  12.   
  13.       
  14.           
  15.         //1,创建字符读取流对象和源相关联。   
  16.         FileReader fr = new FileReader("IO流.txt");  
  17.           
  18.         //2,创建字符输出流对象,明确要存储数据的目的。  
  19.         FileWriter fw = new FileWriter("copy_demo.txt");  
  20.           
  21.         //3,进行读写操作。读取一个,就写入一个。  
  22.         int ch = 0;  
  23.         while((ch=fr.read())!=-1){  
  24.             fw.write(ch);  
  25.         }     
  26.         //4,关闭资源。  
  27.         fw.close();  
  28.         fr.close();  
  29.           
  30.     }  
  31.   
  32. }  

 

缓冲区高效的原因:

流对象的read():是从目的地一次读取一个;

缓冲区的read() :是通过流对象的read( [  ] ) 将一批数据读取到缓冲数组,然后再从数组中一次取一个,所以内存操作要比硬盘操作要高效。

 

 

就在刚才,我们用的高效读取方式,是将字符全部读取到数组中,来达到高效的目的,在Java中,为了提高程序的性能,为我们专门提供了对应的类:

BufferedReader  和BufferedWriter . 接下来,我们将开始学习,这两个对象。

——————————————————————————————————

 

【BufferedReader 】 

从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。可以指定缓冲区的大小,或者可使用默认的大小。大多数情况下,默认值就足够大了。(他的构造方法有两个,一个构造方法使用的是默认大小的缓冲区,另一个构造方法,可以设定自己的缓冲区大小)

他的牛逼之处在于,提供了一个一次读取一行的方法:

readline() ——读取一个文本行。通过下列字符之一即可认为某行已终止:换行 ('\n')、回车 ('\r') 或回车后直接跟着换行。返回值:包含该行内容的字符串,不包含任何行终止符,如果已到达流末尾,则返回 null

下面来具体应用:

 

[java]   view plain copy print ?
  1. import java.io.BufferedReader;  
  2. import java.io.FileReader;  
  3. import java.io.IOException;  
  4.   
  5. public class BuffereReaderDemo {  
  6.   
  7.     public static void main(String[] args) throws IOException {  
  8.         /** 
  9.          * 演示:BufferedReader 
  10.          * 1. 先有字符读取流; 
  11.          * 2. 该类有一个特有方法。readLine(). 一次读一行。 
  12.          */  
  13.         //创建读取流对象 和 文件相关联。  
  14.         FileReader fr = new FileReader("Demo.txt");  
  15.           
  16.         //创建读取缓冲区对象 和 流对象关联对其进行高效 操作;  
  17.         BufferedReader  bufr = new BufferedReader(fr);  
  18.         //简写形式:  
  19.         //BufferedReader bufr = new BufferedReader(new FileReader("Demo.txt"));  
  20.           
  21.         String line = null ;  
  22.         while((line = bufr.readLine()) != null){  
  23.             System.out.println(line);  
  24.         }  
  25.         bufr.close();  
  26.     }  
  27. }  

 

在上述示例中我们应当注意的是:

1.      BufferedReader的简写形式。

BufferedReader bufr = newBufferedReader(new FileReader("Demo.txt"));

 

2. 【readLine().一次读一行的原理

  从缓冲区中取出字符存储到该方法的容器中。当取出的字符是回车符时,就将已经存储的数据作为字符串返回。就是一行数据。

   ReadLine()  调用buf.read()将缓冲区中的数据存储到,自己建立的容器中。ReadLine()建立自己的容器—— StringBuilder 最合适。

 

 

下面我们来学习,BufferedWriter,作为与BufferedReader配套出现的亲兄弟,他的作用,当然显而易见。

[java]   view plain copy print ?
  1. import java.io.BufferedWriter;  
  2. import java.io.FileWriter;  
  3. import java.io.IOException;  
  4.   
  5. public class BufferedWriterDemo {  
  6.   
  7.     public static void main(String[] args) throws IOException {  
  8.   
  9.         //创建一个流对象。  
  10.         FileWriter fw = new FileWriter("buf.txt");  
  11.           
  12.         //为了提高效率。创建缓冲区对象,并和要被提高效率的流相关联。   
  13.         BufferedWriter bufw = new BufferedWriter(fw);  
  14.       
  15.         for(int x=0;x<4; x++){  
  16.             bufw.write(x+"--hahaha");  
  17.             //写入一个换行符。  
  18.             bufw.newLine();  
  19.             //要对缓冲区进行刷新。记住:一般只要使用了缓冲区,就一定要刷新。   
  20.             bufw.flush();  
  21.         }  
  22.           
  23.         //关闭缓冲区。   
  24.         bufw.close();//问:还用关闭fw.close()?不用,因为关闭缓冲区,其实就是在关闭缓冲区关联的流。  
  25.           
  26.     }  
  27. }  

在上述示例中,我们应当注意:

1.      写入一个新行的方法:newline();

2.      每写一次,就要讲缓冲区刷新一次。

3.      在使用完后,关闭缓冲区即可,不需要在关闭流。因为流已经与缓冲区相关联了。

 

 

至此:我们已经学习了IO体系中的四个对象:

 

FileReader  FileWriter  BufferedReader BufferedWriter

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值