Java 你往我来的 IO 流【第 06 篇】 Java 的 字节转字符输出、输入流
字节转字符输出流(可指定编码):OutputStreamWriter
字节转字符输入流(可指定编码):InputStreamReader
关于 Java 常用的 IO 流,整理了 07 篇文章,这是第 06 篇。如果是小白,或者一直没弄清楚 IO 流,请依次把这几篇文章过一遍。当然,高手就不用来我的博客浪费青春了。
小二,上代码:
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.UnsupportedEncodingException;
/*
* 这个类的知识点(字节流转换为字符缓冲流)
* 1、使用 字节输出流 OutputStream 由 OutputStreamWriter 流包装后转 字符输出缓冲流 BufferedWriter 把数据写入文件
* 2、使用 字节输入流 InputStream 由 InputStreamReader 流包装后传转 字符输入缓冲流 BufferedReader 把数据读到内存
*/
public class ByteToChar
{
public static void main(String[] args)
{
// 输出流:将数据从内存写入硬盘文件等,简称:写入输出外界(Writer、OutputStream)
// BufferedWriter 和 Writer 区别:前者缓冲区比后者大,建议使用前者
// 输入流:将数据从硬盘文件等写入内存,简称:读取输入内存(Reader、InputStream)
// BufferedReader 和 Reader 区别:前者缓冲区比后者大,建议使用前者
// 缓冲区作用:在读取数据时,先把数据放到一个缓冲区里(内存),然后再批量进行处理,速度快。如果没有缓冲区,直接操作文件,效率低,性能差。因为读取内存比读取硬盘速度快得多
// 流分两种:字节流和字符流
// 字节流适用场景:主要用来处理字节或二进制对象。如:音频文件、图片、歌曲
// 字符流适用场景:主要用来处理字符或字符串。如:关系到中文(文本)
File file = new File("D:" + File.separator + "temp" + File.separator + "test.txt"); // 在 Windows 下的路径分隔符和 Linux 下的路径分隔符是不一样的,如 Windows 的路径:D:\test.txt,而 Linux 则是:D:/test.txt,所以如果考虑跨平台,最好是用 File.separator 去实现分隔符,它能兼容二者
// 创建目录(这里如果不执行创建,那么必须保证 D 盘上有 temp 目录)
//file.getParentFile().mkdirs();
// 创建文件。这里不用手动创建也可以,因为下面在创建 FileOutputStream 实例时会自动创建
//file.createNewFile();
// 调用字节输出流转字符输出缓冲流方法(把内存数据写入文件)
writeToFile(file);
// 调用字节输入流转字符输入缓冲流方法(把文件数据读取到内存)
readFromFile(file);
}
// 字节输出流转字符输出缓冲流(把内存数据写入文件)
public static void writeToFile(File file)
{
OutputStream os = null; // 为什么不放到 try 里面声明?如果把它放 try 里面声明,那么它的作用域仅限于 try,即只能在 try 里面使用它
OutputStreamWriter osw = null; // 同上
BufferedWriter bw = null; // 同上
try
{
os = new FileOutputStream(file); // FileOutputStream 是 OutputStream(抽象类,不能实例化)的子类
osw = new OutputStreamWriter(os,"UTF-8"); // OutputStreamWriter 是字节流通向字符流的桥梁,需要 FileOutputStream 作参数,这里使用“UTF-8”字符集写入文件。它可以显式指定字符集,或不指定则默认使用平台的字符集
bw = new BufferedWriter(osw); // 字符输出缓冲流,需要 OutputStreamWriter 作为参数
bw.write("101_编程世界_Hello World\r\n102_编程世界_Hello World\r\n103_编程世界_Hello World");
// 字符流使用了缓冲区(字节流没有),将数据写入文件时,数据并没有直接写入文件,而是保存在缓冲区,缓冲区满了之后再写入文件。flush() 方法强制将数据立即写入文件并清空缓冲区
// flush() 不是必须的,但一定要调用 close() 方法
// 如果 flush() 和 close() 方法都不调用,那么数据不能写入文件
bw.flush();
} catch (FileNotFoundException e)
{
e.printStackTrace();
} catch (UnsupportedEncodingException e)
{
e.printStackTrace();
} catch (IOException e)
{
e.printStackTrace();
} finally // finally 作用:无论是否抛出异常,finally 代码块总是会被执行,这里用来释放资源
{
try
{
// 关闭流(如果不关闭流,那么相应的垃圾回收机制就不进行回收,导致垃圾越来越多,浪费资源)
if(bw != null)
{
bw.close();
}
if(osw != null)
{
osw.close();
}
if(os != null)
{
os.close();
}
} catch (IOException e)
{
e.printStackTrace();
}
}
}
// 字节输入流转字符输入缓冲流(把文件数据读取到内存)
public static void readFromFile(File file)
{
InputStream is = null; // 为什么不放到 try 里面声明?如果把它放 try 里面声明,那么它的作用域仅限于 try,即只能在 try 里面使用它
InputStreamReader isr = null; // 同上
BufferedReader br = null; // 同上
try
{
is = new FileInputStream(file); // FileInputStream 是 InputStream(抽象类,不能实例化)的子类
isr = new InputStreamReader(is,"UTF-8"); // InputStreamReader 是字节流通向字符流的桥梁,需要 FileInputStream 作参数,这里使用“UTF-8”字符集读取文件。它可以显式指定字符集,或不指定则默认使用平台的字符集
br = new BufferedReader(isr); // 字符输入缓冲流,需要 InputStreamReader 作为参数
String str = null; // 用来保存每次读取的数据
while((str = br.readLine()) != null) // 每次读取一行,读完则返回 null
{
System.out.println(str);
}
} catch (FileNotFoundException e)
{
e.printStackTrace();
} catch (UnsupportedEncodingException e)
{
e.printStackTrace();
} catch (IOException e)
{
e.printStackTrace();
} finally // finally 作用:无论是否抛出异常,finally 代码块总是会被执行,这里用来释放资源
{
try
{
// 关闭流(如果不关闭流,那么相应的垃圾回收机制就不进行回收,导致垃圾越来越多,浪费资源)
if(br != null)
{
br.close();
}
if(isr != null)
{
isr.close();
}
if(is != null)
{
is.close();
}
} catch (IOException e)
{
e.printStackTrace();
}
}
}
}