字符流
当使用字节流读取文本文件时,可能会有一个小问题。就是遇到中文字符时,可能不会显示完整的字符,那是因为一个中文字符可能占用多个字节存储(如:"学生"两个字占用了四个字节)。所以Java提供一些字符流类,以字符为单位读写数据专门用于处理文本文件。
字符输入流 —— Reader
java.io.Reader抽象类是表示用于读取字符流的所有类的超类,可以读取字符信到内存中。它定义了字符输入流的基本共性功能方法。
public void close() // 关闭此流并释放与此流相关联的任何系统资源。 public int read() // 从输入流读取一个字符。 public int read(char[] chars) // 从输入流中读取一些字符,并将它们存储到字符数组chars中。
FileReader类
java.io.FileReader类是读取字符文件的便利类。构造时使用系统默认的字符编码和默认字节缓冲区。
构造方法:
FileReader(File file) // 刨建一个新的FileReader,给定要读取的File对象。 Filereader( String filename) // 创建一个新的 FileReader,给定要读取的文件的名称。
举例:
字符输入流Reader,类似于字节输入流InputStream。字符输入流一单个字符为单位,字节输入流以一个字节为单位。
/** * 字符输入流使用步骤: * 1、创建FileReader对象,构造方法中绑定要读取的数据源 * 2、使用FileReader对象中的方法read,读取文件 * 3、释放资源 */
import java.io.IOException; import java.io.Reader; import java.io.FileReader; public class DemoFileReader { public static void main(String[] args) throws IOException { // 创建FileReader对象,构造方法中绑定要读取的数据源 FileReader fr = new FileReader("/Users/liyihua/IdeaProjects/Study/src/view/study/demo31/A"); // 使用FileReader对象中的方法read,读取文件 int len = 0; while ((len = fr.read()) != -1) { System.out.println((char) len); } // 释放资源 fr.close(); } }
文件A里面的内容:
控制台输出:
零
一
二
三
四
五
六
七
八
九
十
字符输出流 —— Writer
java.io.Writer抽象类是表示用于写出字符流的所有类的超类,将指定的字符信思写出到目的地。它定义了字节输出流的基本共性功能方法。
public void write(int c) // 写入单个字符。 public void write(char cbuf[]) // 写入字符数组。 abstract public void write(char cbuf[], int off, int len) // 写入字符数组的某一部分,off是数组的开始索引,len是写的字符个数。 public void write(String str) // 写入字符串。 public void write(String str, int off, int len) //写入字符的某一部分。off是字符的开始素,len是写的字符个数。 abstract public void flush() // 刷新该流的缓冲。 abstract public void close() // 关闭此流,但要先刷新它。
FileWriter类
java.io.FileWriter类是写出字符到文件的便利类。构造时使用系统默认的字符编码和默认字节缓冲区。
字符输出流的使用步骤:
- 创建FileWriter对象,构造方法中绑定要写入数据的目的地。
- 使用 FileWriter中的方法write把数据写入到内存缓冲区中(字符转换为字节的过程)
- 使用FileWriter中的方法flush,把内存缓冲区中的数据刷新到文件中。
- 释放资源(会先把内存冲区中的数据刷新到文件中)。
import java.io.IOException; import java.io.FileWriter; public class DemoFileWriter { public static void main(String[] args) throws IOException { // 1.创建FileWriter对象,构造方法中绑定要写入数据的目的地。 FileWriter fw = new FileWriter("/Users/liyihua/IdeaProjects/Study/src/view/study/demo31/test.txt"); // 2.使用FileWriter中的方法write把数据写入到内存缓冲区中(字符转换为字节的过程)。 fw.write("Hello FileWriter"); // 3.使用FileWriter中的方法flush,把内存缓冲区中的数据刷新到文件中。 fw.flush(); // 4.释放资源(会先把内存冲区中的数据刷新到文件中)。 fw.close(); } }
关闭和刷新
因为内置缓冲区的原因,如果不关闭输出流,无法写出字符到文件中。但是关闭的流对象,是无法继续写出数据的。如果我们既想写出数据,又想继续使用流,就需要flush方法了。
- flush:刷新绶冲区,流对象可以继续使用。
- close:先刷新绶冲区,然后通知系统释放资源。流对象不可以再被使用了。
如上个例子,在使用了flush()方法后可以继续使用write方法,而在使用了close()方法后,就不可以继续使用write方法了,否则在运行期会抛出错误。
写出其他数据
write(char cbuf[])方法和write(char cbuf[], int off, int len)方法,每次可以写出字符数组的数据,即将字符数组中的数据写入到文件中去。
import java.io.FileWriter; import java.io.IOException; public class Demo1FileWriter { public static void main(String[] args) throws IOException { FileWriter fw = new FileWriter("/Users/liyihua/IdeaProjects/Study/src/view/study/demo31/write1.txt"); char[] chars = "编程路上的小学生".toCharArray(); fw.write(chars); char[] chars1 = {97, 98, 99}; fw.write(chars1, 0, 2); fw.flush(); fw.close(); } }
运行结果:
续写与换行
操作类似于FileOutputStream
import java.io.FileWriter; import java.io.IOException; public class DemoAppendAddLindWrite { public static void main(String[] args) throws IOException { FileWriter fw = new FileWriter("/Users/liyihua/IdeaProjects/Study/src/view/study/demo31/test2.txt", true); fw.write("工程师"); fw.write(" "); fw.write("程序员"); fw.flush(); fw.close(); } }
程序运行两次:
IO异常的处理
import java.io.FileWriter; import java.io.IOException; public class DemoFileWriteTryCatch { public static void main(String[] args) { // 创建FileWriter并赋值Null FileWriter fw = null; try { // 创建FileWriter对象 fw = new FileWriter("/Users/liyihua/IdeaProjects/Study/src/view/study/demo31/test3.txt", true); // 写入数据 fw.write("处理异常测试输入1"); fw.write(" "); fw.write("处理异常测试输入2"); // 刷新 fw.flush(); } catch (IOException e) { // 自己处理异常 System.out.println(e.getMessage()); } finally { // 无论咋滴,最后都释放资源 try { // 关闭,当且仅当fw != null的时候 assert fw != null; fw.close(); } catch (IOException e) { // 自己处理异常 System.out.println(e.getMessage()); } } } }
运行两次结果:
在JDK7后,可以直接在try后创建流对象,而不用写finally语句块,最后也会释放资源(自动释放资源)。
格式:
try (定义流对象1; 定义流对象2.....) { 可能会出现异常的代码块 } catch (异常变量 变量名) { 异常的逻辑 }
举例(文件复制):
// 没使用try...catch import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; /** * 文件复制的步骤: * 1.创建一个字节输入流对象,构造方法中绑定要读取的数据源。 * 2.创建一个字节输出流对象,构造方法中绑定要写入的目的地。 * 3.使用字节输入流对象中的方法read读取文件。 * 4.使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中。 * 5.释放资源。 */ public class Demo01CopyFile { public static void main(String[] args) throws IOException { // 创建一个字节输入流对象,构造方法中绑定要读取的数据源。 FileInputStream fis = new FileInputStream("/Users/liyihua/IdeaProjects/Study/src/view/study/demo30/666.jpg"); // 创建一个字节输出流对象,构造方法中绑定要写入的目的地。 FileOutputStream fos = new FileOutputStream("/Users/liyihua/IdeaProjects/Study/src/view/study/demo30/999.jpg"); // 使用字节输入流对象中的方法read读取文件。 int len = 0; while ((len = fis.read()) != -1) { // 使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中。 fos.write(len); } // 释放资源。 fis.close(); fos.close(); } }
// 使用try...catch import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; public class DemoJDK7 { public static void main(String[] args) { try ( FileInputStream fis = new FileInputStream("/Users/liyihua/IdeaProjects/Study/src/view/study/demo31/666.jpg"); FileOutputStream fos = new FileOutputStream("/Users/liyihua/IdeaProjects/Study/src/view/study/demo31/999.jpg"); ) { int len = 0; while ((len = fis.read()) != -1) { fos.write(len); } } catch (IOException e) { System.out.println(e.getMessage()); } } }
运行两个程序结果都一样,将文件666.jpg复制为999.jpg