JavaEE 初阶篇-深入了解 I/O 流(FileInputStream 与 FileOutputStream 、Reader 与 Writer)

ops/2024/12/22 2:56:30/

🔥博客主页: 【小扳_-CSDN博客】
❤感谢大家点赞👍收藏⭐评论✍
 

文章目录

        1.0 I/O 流概述

        2.0 文件字节输入流(FileInputStream)

        2.1 创建 FileInputStream 对象

        2.2 读取数据

        2.3 关闭流

        3.0 文件字节输出流(FileOutputStream)

        3.1 创建 FileOutputSrteam 对象

        3.2 写入数据

        3.3 写入字节数组

        3.3 关闭流

        4.0 用字节输入流与字节输出流来实现文件复制

        5.0 释放资源的方式

        6.0 文件字符输入流(FileReader)

        6.1 创建 FileReader 对象

        6.2 读取字符

        6.3 读取字符数组

        7.0 文件字符输出流(FileWriter)

        7.1 创建 FileWriter 对象

        7.2 写入字符


        1.0 I/O 流概述

        I/O 流是 Java 中用于处理输入和输出的机制,它提供了一种统一的方式来处理不同来源和目的地的数据。在 Java 中,I/O 流主要用于与文件、网络、内存等进行数据的读取和写入操作。

        I 指 Input,称为输入流:负责把数据读到内存中。

        O 指 Output,称为输出流:负责把数据从内存中写到磁盘或者网络等等。

        I/O 流可以分为字节流和字符流两种类型。字节流以字节为单位读写数据,适用于处理二进制文件或字节数据;字符流以字符为单位读写数据,适用于处理文本文件或字符数据。

        2.0 文件字节输入流(FileInputStream)

        FileInputStream 是 Java 中用于从文件中读取字节数据的输入流类。它继承 InputStream 类,提供了一些方法来读取文件中的字节数据。

        2.1 创建 FileInputStream 对象

        可以使用 FileInputStream 的构造函数来创建对象,需要传入要读取的文件路径或者文件对象作为参数。

代码如下:

java">FileInputStream fis = new FileInputStream("path/to/file.txt");

         简单来说,创建了 FileInputStream 对象相当于创建了连通文件与内存之间的管道,进行字节流的流通。但是,该管道只能从文件流到内存中。

        2.2 读取数据

        FileInputStream 提供了 read() 方法来读取文件中的字节数据。每次调用 read() 方法会读取一个字节,并返回读取的字节数据(以 int 类型表示,范围为 0 到 255),如果已到达文件末尾,则返回 -1。

举个例子:

        

        将该文件中的内容读取到内存中。 

代码如下:

java">import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;public class demo1 {public static void main(String[] args) throws IOException {InputStream inputStream = new FileInputStream("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");//每次读取一个字节int n;//直到返回-1,就意味这已经将全部内容读取完毕了while ( (n = inputStream.read()) != -1 ){//因为读取出来的是字节,读出来的都是整数,那么可以将其转换为字符System.out.print((char) n);}//关闭资源inputStream.close();}
}

运行结果:

        除了单个字节的读取, FileInputStream 还提供了 read(byte[] b) 方法来一次性读取多个字节到指定的字节数组中,其中返回值为读取的字节个数。

        简单来说,可以将 b 数组当作成一个容器,那么 b 数组一下子可以从文件中装指定大小个字节,并且返回 b 容器中装了多少了字节,一旦返回值为 -1 ,则代表已经将文件中的内容读取完毕了。

        该方法的效率远比一个一个字节读取到内存中的方法高效很多。

代码如下:

java">import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;public class demo2 {public static void main(String[] args) throws IOException {InputStream inputStream = new FileInputStream("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");//指定容器大小为 1024 个字节(1kb)byte[] b = new byte[1024];int n ;while ((n = inputStream.read(b)) != -1){//为了更好的观察,转换成字符串,不可能每一次都恰好可以装满1kb,只解析 0 到 n 个字节。String s = new String(b,0,n);System.out.println(s);}//关闭流inputStream.close();}
}

运行结果:

注意事项:

        1)使用 FileInputStream 每次读取一个字节,读取性能较差,并且读取汉字输出会乱码。

        2)使用 FileInputStream 每次读取多个字节,读取性能得到提升,但读取汉字输出还是会乱码。

使用字节流读取中文,如何保证输出不乱码,怎么解决?

        定义一个与文件一样大小的字节数组,一次性读取完文件的全部字节。

代码如下:

java">import java.io.*;public class demo3 {public static void main(String[] args) throws IOException {File file = new File("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");InputStream inputStream = new FileInputStream(file);byte[] b = new byte[(int)file.length()];inputStream.read(b);String s = new String(b);System.out.println(s);inputStream.close();}
}

运行结果:

        此外官方也提供了相对应的方法 readAllBytes()

代码如下:

java">import java.io.*;public class demo4 {public static void main(String[] args) throws IOException {File file = new File("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");InputStream inputStream = new FileInputStream(file);byte[] b = inputStream.readAllBytes();String s = new String(b);System.out.println(s);}
}

        效果是跟自己实现的代码时相同的。

运行结果:

直接把文件数据全部读取到一个字节数组可以避免乱码,是否存在问题?

        如果文件过大,创建的字节数组也会过大,可能引起内存溢出。

        无论用官方实现的代码还是自己实现的代码,面对文件中的内容较大的时候,用该方法就很不现实了,因为磁盘肯定比内存空间大很多。读取文本内容更适合用字符流。而字节流适合坐数据的转移,如:文件复制等。

        2.3 关闭流

        在读取完文件数据后,应该及时关闭 FileInputStream 对象以释放资源。可以使用 close() 方法来关闭流。

        以上代码已经出现过了。

java">fis.close();

        3.0 文件字节输出流(FileOutputStream)

        FileOutputStream 是 Java 中用于向文件中写入字节数据的输出流类。它继承 OutputStream类,提供了一些方法来向文件中写入字节数据。

        3.1 创建 FileOutputSrteam 对象

        可以使用 FileOutStream 的构造函数来创建对象,需要存入要写入的文件路径作为参数。如果文件不存在,会自动创建新文件;如果文件已存在,会覆盖原有内容,也可以给构造器中再传入一个参数 true ,以支持继续追加,不会覆盖原有的内容。

代码如下:

java">import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.OutputStream;public class demo5 {public static void main(String[] args) throws FileNotFoundException {OutputStream outputStream = new FileOutputStream("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");}
}

        简单来说:创建 FileOutputStream 对象相当于创建了连接内存与文件的通道,只是该通道只能将字节流从内存流到文件中。

        3.2 写入数据

        FileOutputStrea 提供了 write(int b) 方法来向文件中写入一个字节数据。可以将一个整数作为参数传递,只会取低 8 位字节写入文件。

代码如下:

java">import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;public class demo5 {public static void main(String[] args) throws IOException {OutputStream outputStream = new FileOutputStream("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");outputStream.write(97);outputStream.write(98);outputStream.write(99);outputStream.write(100);outputStream.write(101);outputStream.write(102);outputStream.write(103);outputStream.write(104);outputStream.close();}
}

运行结果:

        3.3 写入字节数组

        除了单个字节的写入,FileOutputStream 还提供了 write(byte[] b) 方法来一次性写入整个字节数组中的数据。

代码如下:

java">import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;public class demo6 {public static void main(String[] args) throws IOException {OutputStream outputStream = new FileOutputStream("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");String str = "土豆土豆我是地瓜";outputStream.write(str.getBytes());outputStream.close();}
}

运行结果:

        如果不想覆盖原有的内容,那么在构造 FileOutputSrteam 对象的时候,构造方法再传入一个参数 true ,代表允许追加内容。

代码如下:

java">import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;public class demo6 {public static void main(String[] args) throws IOException {OutputStream outputStream = new FileOutputStream("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text",true);String str = "土豆土豆我是地瓜";outputStream.write(str.getBytes());outputStream.close();}
}

运行结果:

        再次强调,是以一个字节输出或者输出字节数组到文件中。

        3.3 关闭流

        在读取完文件数据后,应该及时关闭 FileOutputStream 对象以释放资源。可以使用 close() 方法来关闭流。

        以上代码已经出现过了。

java">fis.close();

        4.0 用字节输入流与字节输出流来实现文件复制

        思路:先将要读取的文件创建通道,将文件中字节流流到内存中,再将内存中的字节流流到指定的文件中。即创建文件 ==> 读取数据 ==> 写入数据 。

代码如下:

java">import java.io.*;public class demo7 {public static void main(String[] args) throws Exception {copy("D:\\照片\\201651723362635996.jpg","D:\\software\\aaa.jpg");}public static void copy(String path, String obj) throws Exception {if (path == null || obj == null){return;}InputStream inputStream = new FileInputStream(path);OutputStream outputStream = new FileOutputStream(obj);byte[] b = new byte[3];int n;while ((n = inputStream.read(b)) != -1){outputStream.write(b,0,n);}outputStream.close();inputStream.close();}
}

运行结果:

字节流非常适合做一切文件的复制操作:

        任何文件的底层都是字节,字节流复制,是一字不漏的转移完全部字节,只要复制后的文件格式一致就没问题。

        5.0 释放资源的方式

        在 Java 中释放资源通常是指关闭文件流、数据库连接、网络连接等资源,以避免资源泄露和浪费。以下是几种常见的释放资源的方式:

        1)使用 try-with-resources:Java 7 引入了 try-with-resources 语句,可以自动关闭实现了 AutoCloseable 接口的资源。在 try-with-resources 语句中创建的资源会在代码块执行完毕后自动关闭,无需手动调用 close() 方法。

java">try (FileInputStream fis = new FileInputStream("file.txt")) {// 读取文件内容
} catch (IOException e) {// 异常处理
}

        2)手动关闭资源:如果无法使用 try-with-resources,需要手动关闭资源。在不再需要资源时,通过调用资源的 close() 方法来关闭资源。

java">FileInputStream fis = null;
try {fis = new FileInputStream("file.txt");// 读取文件内容
} catch (IOException e) {// 异常处理
} finally {if (fis != null) {try {fis.close();} catch (IOException e) {// 异常处理}}
}

        6.0 文件字符输入流(FileReader)

        FileReader 是 Java 中用于读取字符数据的输入流,继承自 Reader 类。FileReader 可以用来读取文本文件中的字符数据。

        6.1 创建 FileReader 对象

        使用 FileReader,首先需要创建 FileReader 对象并指定要读取的文件路径或者文件对象。

代码如下:

java">FileReader fr = new FileReader("file.txt");

        6.2 读取字符

        FileReader 提供了 read() 方法来读取单个字符。它会返回一个整数,表示读取的字符的 Unicode 编码。读取到文件末尾时返回 -1。

代码如下:

java">import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;public class demo8 {public static void main(String[] args) {try (Reader reader = new FileReader("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text");){int n;while ((n = reader.read()) != -1){System.out.print((char) n);}}catch (IOException e){e.printStackTrace();}}
}

运行结果:

        6.3 读取字符数组

        FileReader 还提供了 read(char[] cbuf )方法来一次性读取多个字符,并将其存储到字符数组中。读取到文件末尾时返回 -1。返回值是读取的字符个数。

代码如下:

java">import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;public class demo9 {public static void main(String[] args) {try (Reader reader = new FileReader("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text")){char[] chars = new char[3];int n;while ((n = reader.read(chars)) != -1){String s = new String(chars,0,n);System.out.print(s);}}catch (IOException e){e.printStackTrace();}}
}

运行结果:

        7.0 文件字符输出流(FileWriter)

        FileWriter 是 Java 中用于写入字符数据的输出流,继承自 Writer 类。FileWriter 可以用来向文件中写入字符数据。

        7.1 创建 FileWriter 对象

        要使用 FileWriter ,首先需要创建 FileWriter 对象并指定要写入的文件路径或者文件对象。可以指定是否追加数据到文件末尾。

代码如下:

java">FileWriter fw = new FileWriter("file.txt");

        简答来说,创建 FileWriter 对象相当于创建连接了内存与文件的通道,只是该通道只能将字符流从内存流到文件中。

        7.2 写入字符

        FileWriter 提供了 write(int c) 方法来写入单个字符,以及 write(String str) 方法来写入字符串,还可以写入字符数组。

代码如下:

java">public class demo10 {public static void main(String[] args) {try (Writer writer = new FileWriter("D:\\software\\code\\2023_java\\2023_java_code\\code_24_4_21\\src\\MyFile\\MyText.text")){writer.write(97);//换行writer.write("\r\n");writer.write("土豆土豆我是地瓜");//换行writer.write("\r\n");writer.write('c');//换行writer.write("\r\n");String s = "地瓜地瓜我是土豆";writer.write(s.toCharArray());}catch (IOException e){e.printStackTrace();}}
}

运行结果:

注意事项:

        字符输出流写出数据后,必须刷新流,或者关闭流,写出去的数据才能生效。

        写出数据时,数据会先被写入到内存缓冲区中,而不是直接写入到文件中。这是为了提高写入效率,减少频繁地访问磁盘的开销。

        因此,为了确保写出去的数据能够及时生效,你可以选择调用 flush() 方法来刷新缓冲区,或者调用 close() 方法来关闭流。这样可以保证数据被写入到文件中,而不会遗失或被丢弃。


http://www.ppmy.cn/ops/14460.html

相关文章

css中新型的边框设置属性border-inline

一、概念与背景 border-inline 是 CSS Logical Properties and Values 模块中的一个属性,用于控制元素在流内(inline)方向上的边框。该模块旨在提供与书写模式(writing mode)无关的布局和样式描述方式,使得…

Django模型继承之多表继承

在Django模型继承中,支持的第二种模型继承方式是层次结构中的每个模型都是一个单独的模型。每个模型都指向分离的数据表,并且可以被独立查询和创建。在继承关系中,子类和父类之间通过一个自动创建的OneToOneField进行连接。示例代码如下&…

Java23种设计模式-创建型模式之抽象工厂模式

抽象工厂模式(Abstract Factory Pattern)是一种创建型设计模式,它用于创建相关或相互依赖对象的一组,而无需指定其具体的类。这种模式特别适用于产品族的情况,即一组相互关联的产品对象。 存在四种角色: 角色1:抽象工…

C++ 继承

C/C总述:Study C/C-CSDN博客 目录 基类&派生类 访问控制和继承 继承基类成员访问方式的变化 改变访问权限 继承中的作用域 派生类的默认成员函数 继承的静态成员与友元 多继承与菱形继承 虚拟继承 继承与组合 面向对象程序设计中最重要的一个概念是…

2、选择什么样的机器人本体

如果说世界是物质的,那么应该先制造出机器人的本体,再让她产生灵魂。如果是精神的呢,世界是无中生有的呢,那就先在仿真中研究算法吧。 而我比较崇尚初中哲学的一句话,世界是物质的,物质是运动的&am…

【随想录】Day31—第八章 贪心算法 part01

目录 题目1: 455. 分发饼干1- 思路2- 题解⭐分发饼干 ——题解思路 题目2: 摆动序列1- 思路2- 题解⭐摆动序列 ——题解思路 题目3: 最大子数组和1- 思路2- 题解⭐ 最大子数组和 ——题解思路 题目1: 455. 分发饼干 题目链接:455. 分发饼干 1- 思路 贪心的思路&am…

Scala详解(6)

Scala 集合 字符串 Scala中字符串同样分为可变字符串和不可变字符串,不可变字符串使用String来定义,可变字符串使用的是StringBuilder来定义 package com.fesco.string ​ object StringDemo { ​def main(args: Array[String]): Unit { ​// 可变字符…

【03-掌握Scikit-learn:深入机器学习的实用技术】

文章目录 前言数据预处理缺失值处理数据缩放特征选择模型训练参数调整模型评估总结前言 经过了对Python和Scikit-learn的基础安装及简单应用,我们现在将更深入地探究Scikit-learn的实用技术,以进一步提升我们的数据科学技能。在本文中,我们将涵盖数据预处理、特征选择、模型…