1. 什么是流
Java中的流是对字节序列的抽象,我们可以想象有一个水管,只不过现在流动在水管中的不再是水,而是字节序列。和水流一样,Java中的流也具有一个“流动的方向”,通常可以从中读入一个字节序列的对象被称为输入流;能够向其写入一个字节序列的对象被称为输出流
2. 字节流
Java中的字节流处理的最基本单位为单个字节,它通常用来处理二进制数据。Java中最基本的两个字节流类是InputStream和OutputStream,它们分别代表了一组基本的输入字节流和输出字节流。InputStream类与OutputStream类均为抽象类,我们在实际使用中通常使用Java类库中提供的它们的一系列子类。
3. 字符流
Java中的字符流处理的最基本的单元是Unicode码元(大小2字节),它通常用来处理文本数据。所谓Unicode码元,也就是一个Unicode代码单元,范围是0×0000~0xFFFF。在以上范围内的每个数字都与一个字符相对应,Java中的String类型默认就把字符以Unicode规则编码而后存储在内存中。然而与存储在内存中不同,存储在磁盘上的数据通常有着各种各样的编码方式。使用不同的编码方式,相同的字符会有不同的二进制表示。
4. 字符流与字节流的区别
经过以上的描述,我们可以知道字节流与字符流之间主要的区别体现在以下几个方面:
字节流操作的基本单元为字节;字符流操作的基本单元为Unicode码元。
字节流默认不使用缓冲区;字符流使用缓冲区。
字节流通常用于处理二进制数据,实际上它可以处理任意类型的数据,但它不支持直接写入或读取Unicode码元;字符流通常处理文本数据,它支持写入及读取Unicode码元。
示例一:文件读取和写入(支持编码选择、文件内容追加)
/** * 读取文件文本内容。 * * @param filePath 文件路径 * @param charset 字符编码格式 * @throws Exception */ public static String readAllText(String filePath, String charset) throws Exception { StringBuilder sb = new StringBuilder(); try ( InputStreamReader reader = new InputStreamReader(new FileInputStream(filePath), charset); BufferedReader br = new BufferedReader(reader) ) { String line; while ((line = br.readLine()) != null) { sb.append(line); sb.append(SystemCharUtils.getNewLine()); } } catch (Exception e) { throw e; } return sb.toString(); } /** * 将字符串追加到文件。 * * @param str 待写入字符串 * @param filePath 文件路径(文件不存在则自动创建) * @param charset 字符编码 * @throws Exception */ public static void appendFile(String str, String filePath, String charset) throws Exception { // 创建新文件对象。 File file = new File(filePath); //自动创建目录。 if (!file.getParentFile().exists()) { if (!file.getParentFile().mkdirs()) { throw new RuntimeException("文件目录创建失败:" + file.getParentFile().getAbsolutePath()); } } //如果文件不存在则自动创建。 if (!file.exists()) { file.createNewFile(); } try ( OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(file, true), charset); BufferedWriter bw = new BufferedWriter(writer); ) { bw.write(str); } catch (Exception e) { throw e; } }
示例二:对象序列化到文件流 及 从文件流反序列化为对象
注意:实体对象要实现Serializable接口。
/** * 序列化文件路径。 */ String localFilePath = SystemCharUtils.getUserDir() + SystemCharUtils.getFileSeparator() + "data" + SystemCharUtils.getFileSeparator() + "test" +SystemCharUtils.getFileSeparator() + "testData.data"; /** * 从文件流反序列化为对象。 * * @param filePath 文件路径 * @throws Exception */ public static <T> T getObject(String filePath) throws Exception { try ( ObjectInputStream inputStream = new ObjectInputStream(new FileInputStream(filePath)); ) { return (T) inputStream.readObject(); } catch (Exception e) { throw e; } } /** * 将对象序列化到文件流。 * * @param obj 需要序列化的对象 * @param filePath 文件路径 * @throws Exception */ public static void saveObject(Object obj, String filePath) throws Exception { // 创建新文件对象。 File file = new File(filePath); //自动创建目录。 if (!file.getParentFile().exists()) { if (!file.getParentFile().mkdirs()) { throw new RuntimeException("文件目录创建失败:" + file.getParentFile().getAbsolutePath()); } } //如果文件不存在则自动创建。 if (!file.exists()) { file.createNewFile(); } try ( ObjectOutputStream outputStream = new ObjectOutputStream(new FileOutputStream(file)); ) { outputStream.writeObject(obj); } catch (Exception e) { throw e; } }