java-IO流笔记

时间:2025-03-30 18:16:28

一、I/O 流基础概念

  1. 流(Stream)的本质

    • 数据流动的抽象:将数据的输入/输出操作抽象为“流”,类似水流从源头到目的地的过程。
    • 方向性输入流(InputStream/Reader) 读取数据,输出流(OutputStream/Writer) 写入数据。
  2. 核心分类

    分类依据 类别 特点
    数据单位 字节流(Byte Stream) 处理二进制数据(如图片、视频)
    字符流(Char Stream) 处理文本数据(自动处理字符编码)
    数据流动方向 输入流 从数据源读取到程序
    输出流 从程序写入到目的地
    功能层级 节点流 直接操作数据源(如文件、网络)
    处理流(包装流) 对节点流封装,增强功能(如缓冲)

二、核心类层次结构

1. ​字节流​(InputStream / OutputStream
InputStream
├─ FileInputStream       // 文件字节输入流
├─ ByteArrayInputStream  // 字节数组输入流
├─ FilterInputStream     // 装饰器模式基类
   ├─ BufferedInputStream // 缓冲字节输入流(提升性能)
   ├─ DataInputStream     // 读取基本数据类型(如int, double)
OutputStream
├─ FileOutputStream      // 文件字节输出流
├─ ByteArrayOutputStream // 字节数组输出流
├─ FilterOutputStream
   ├─ BufferedOutputStream
   ├─ DataOutputStream
1.1 InputStream 读
1.1.1 FileInputStream

InputStream 字节读取流父类 抽象类

FileInputStream

read() : 每次读取一个字节 返回值为读取的内容 ASCII码 如果读取到文件的末尾 返回值为-1 close() 关闭字节流

package com.atguigu.test4;


import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 14:12
 *  InputStream 字节读取流父类 抽象类
 *  FileInputStream
 *
 *  read() : 每次读取一个字节 返回值为读取的内容 ASCII码 如果读取到文件的末尾 返回值为-1
 *
 *
 *  close() 关闭字节流
 */
public class TestFileInputStream1 {
    public static void main(String[] args) {
        // 使用相对路径创建File对象
        File file = new File("A.txt"); // D:\\ideaProjects\\java0724day19\\
        FileInputStream inputStream = null;
        try {
            // 创建字节读取流对象 参数为file对象
            inputStream = new FileInputStream(file);
            // 读取文件 读取一个字节
            int readData = inputStream.read();
            // 打印 因为读取内容的返回值为int类型 所以需要强转为char类型
            System.out.println("readData = " + (char)readData);
            // 读取文件 读取一个字节
            readData = inputStream.read();
            // 打印 因为读取内容的返回值为int类型 所以需要强转为char类型
            System.out.println("readData = " + (char)readData);
            // 使用循环多次读取 循环停止的条件为 读取内容返回值为-1
            while((readData = inputStream.read()) != -1){
                System.out.println((char)readData); // 强制类型转换
            }
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }finally{
            // 做非null判断
            if(inputStream != null){
                try {
                    inputStream.close(); // 关闭资源
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

read(byte [] data) : 每次读取一个字节数组 返回值为读取字节的个数 读取的内容保存在byte数组中

如果读取到文件的末尾 返回值为-1

package com.atguigu.test4;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 14:26
 *  read(byte [] data) : 每次读取一个字节数组 返回值为读取字节的个数 读取的内容保存在byte数组中
 *  如果读取到文件的末尾 返回值为-1
 */
public class TestFileInputStream2 {
    public static void main(String[] args) throws IOException {
        FileInputStream inputStream = new FileInputStream("A.txt");
        // 定义数组 长度为15 表示每次读取15个字节
        byte [] data = new byte[15];

        // 定义变量表示实际读取的字节的个数
        int readCount = -1;
        // 循环读取 每次读取到的内容保存在byte数组中 读取到的个数 保存在 readCount中
        while((readCount = inputStream.read(data)) != -1) {
            // 解析byte数组 转换为字符串
            // 第一个参数 byte数组
            // 第二个参数 从0开始转换
            // 第三个参数 转换的数量(即读取到几个 就转换几个字节)
            System.out.println(new String(data, 0, readCount));

        }

        inputStream.close();

        System.out.println("程序结束");



    }
}

使用字节流读取中文

available() 返回当前字节流可读字节数

中文占几个字节?

UTF-8 占3个字节

GBK 占2个字节

package com.atguigu.test4;


import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 14:38
 *  使用字节流读取中文
 *  available() 返回当前字节流可读字节数
 *  中文占几个字节?
 *      UTF-8 占3个字节
 *      GBK 占2个字节
 *
 */
public class TestFileInputStream3 {
    public static void main(String[] args) {
        FileInputStream inputStream = null;
        try {
            inputStream = new FileInputStream("A.txt");
            int available = inputStream.available();

            System.out.println("可读字节数 = " + available);

            // 使用可读字节数来定义数组长度 避免拆分中文 导致乱码
            byte [] data = new byte[inputStream.available()];

            int readCount = -1;

            // 循环读取
            while((readCount = inputStream.read(data)) != -1){
                System.out.println(new String(data,0,readCount)); // 转换字符串
            }
        } catch (FileNotFoundException e) {
           e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }finally{
            // 关闭资源
        }

    }
}
2.2 OutputStream 写
2.2.1 FileOutputStream

OutputStream

FileOutputStream

new FileOutputStream(String fileName) 传入字符串

new FileOutputStream(File file) 传入文件对象

new FileOutputStream(File file , boolean append) true表示追加 默认不写 或者 false 表示覆盖源文件

使用字节写入流写入文件 如果文件不存在 则自动创建

write(int data) 写入单个字节

write(byte [] data) 写入字节数组

close() 关闭写入字节流

package com.atguigu.test5;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 15:16
 *  OutputStream
 *  FileOutputStream
 *
 *  new FileOutputStream(String fileName) 传入字符串
 *  new FileOutputStream(File file) 传入文件对象
 *  new FileOutputStream(File file , boolean append)   true表示追加 默认不写 或者 false 表示覆盖源文件
 *
 *  使用字节写入流写入文件 如果文件不存在 则自动创建
 *
 *
 *  write(int data) 写入单个字节
 *  write(byte [] data) 写入字节数组
 *  close() 关闭写入字节流
 *
 *
 *
 */
public class TestFileOutputStream1 {
    public static void main(String[] args) {
        FileOutputStream outputStream = null;
        try {
            outputStream = new FileOutputStream("B.txt");

            outputStream.write(65);
            outputStream.write(66);
            outputStream.write(67);
            outputStream.write(68);
            outputStream.write(69);

        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }finally{
            if(outputStream != null){
                try {
                    outputStream.close();
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            }

        }
    }
}

write(byte [] data) 写入字节数组

package com.atguigu.test5;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 15:20
 *   write(byte [] data) 写入字节数组
 */
public class TestFileOutputStream2 {
    public static void main(String[] args) throws IOException {
        FileOutputStream outputStream = new FileOutputStream("C.txt",true);

        byte [] data = {101};

        outputStream.write(data);

        outputStream.close();

    }
}

使用字节写入流写入中文

package com.atguigu.test5;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 15:23
 *  使用字节写入流写入中文
 */
public class TestFileOutputStream3 {
    public static void main(String[] args) throws IOException {
        FileOutputStream outputStream = new FileOutputStream("D.txt",true);

        String str = "世界你好 hello world 666";

        byte[] bytes = str.getBytes();

        outputStream.write(bytes);

        outputStream.close();

    }
}
2. ​字符流​(Reader / Writer

Reader 

  • Reader 字符读取流 父类 抽象类
  • InputStreamReader
  • read() 每次读取一个字符 返回值为读取的内容 如果读取到文件末尾 返回值为-1
  • close() 关闭资源

InputStreamReader

Reader
├─ InputStreamReader     // 字节流转字符流(需指定编码)
   └─ FileReader         // 文件字符输入流(默认平台编码)
├─ BufferedReader        // 缓冲字符输入流
Writer
├─ OutputStreamWriter    // 字符流转字节流(需指定编码)
   └─ FileWriter         // 文件字符输出流(默认平台编码)
├─ BufferedWriter        // 缓冲字符输出流

package com.atguigu.test6;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 15:30
 * Reader 字符读取流 父类 抽象类
 * InputStreamReader
 * <p>
 * read() 每次读取一个字符 返回值为读取的内容 如果读取到文件末尾 返回值为-1
 * close() 关闭资源
 */
public class TestInputStreamReader1 {
    public static void main(String[] args) {
        FileInputStream fileInputStream = null;
        InputStreamReader reader = null;
        try {
            fileInputStream = new FileInputStream("A.txt"); // 创建字节读取流对象
            reader = new InputStreamReader(fileInputStream); // 创建字符读取对象

            int data = reader.read(); // 每次读取一个字符

            System.out.println((char)data); // 打印 强转为char类型

            data = reader.read();

            System.out.println((char)data);

            while((data = reader.read())!= -1){ // 循环读取 返回值为-1表示读取到文件末尾
                System.out.println((char)data); // 强转为char类型
            }



        } catch (FileNotFoundException e) {
            throw new RuntimeException(e);
        } catch (IOException e) {
            throw new RuntimeException(e);
        }finally{
            // 关闭资源  先用后关
            try{
                if(reader != null){ // 非null判断 
                    reader.close(); // 关闭资源
                }
                if(fileInputStream != null){// 非null判断 
                    fileInputStream.close();// 关闭资源
                }
            }catch(IOException e){
                e.printStackTrace();
            }
        }

    }
}

        read(char [] data) 每次读取一个字符数组 返回值为读取字符的个数 读取的内容保存在数组中 读取到末尾 返回值为-1

package com.atguigu.test6;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 15:38
 *  read(char [] data) 每次读取一个字符数组 返回值为读取字符的个数 读取的内容保存在数组中  读取到末尾 返回值为-1
 */
public class TestInputStreamReader2 {
    public static void main(String[] args) throws IOException {
        InputStreamReader reader = new InputStreamReader(new FileInputStream("A.txt"));

        char [] data = new char[5];

        int readCount = -1;

        while((readCount = reader.read(data)) != -1){
//            System.out.println(data); //
            // 将字符数组 转换为String对象
            // 第一个参数为要转换的字符数组
            // 第二个参数为0 表示从数组中的第一个位置开始转换
            // 第三个参数 为转换的个数 readCount表示读取了多少 就转换多少
            System.out.println(new String(data, 0, readCount));
        }

        byte [] b = {1,2,3,4,5};
        System.out.println(b);

        char [] c = {'a','b','c'};
        System.out.println(c); // println方法单独对char数组作为参数 做了操作 直接遍历 拼接 再 打印


    }
}

使用InputStreamReader解决读取中文乱码问题

如果文件的编码格式 和 本地平台默认读取文件的编码格式不一致

将产生乱码 可以使用InputStreamReader类构造方法 传入 编码格式 来指定编码

package com.atguigu.test6;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/19 15:47
 *  使用InputStreamReader解决读取中文乱码问题
 *
 *  如果文件的编码格式 和 本地平台默认读取文件的编码格式不一致
 *  将产生乱码 可以使用InputStreamReader类构造方法 传入 编码格式 来指定编码
 *
 */
public class TestInputStreamReader3 {
    public static void main(String[] args) throws IOException {
        FileInputStream fileInputStream = new FileInputStream("C:\\Users\\WHD\\Desktop\\0324.txt");

        // 第二个参数为 指定读取文件的编码格式 
        InputStreamReader inputStreamReader = new InputStreamReader(fileInputStream,"GBK");

        char [] data = new char[10];

        int readCount = -1;

        while((readCount = inputStreamReader.read(data)) != -1){
            System.out.println(new String(data,0,readCount)); // 解析字符串
        }

        inputStreamReader.close();

        fileInputStream.close();





    }
}

FileReader

Reader

InputStreamReader

FileReader : 只能按照本地平台默认的编码格式读取数据 不能单独指定数据

read() 每次读取一个字符 返回值为读取内容的ASCII码 或者 Unicode编码 十进制 如果读取到末尾 返回-1

read(char [] data) 每次读取一个字符数组 返回值为读取字符的个数 如果读取到末尾 返回-1

close() 关闭资源

package com.atguigu.test1;

import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/21 9:09
 * Reader
 * InputStreamReader
 * FileReader   : 只能按照本地平台默认的编码格式读取数据 不能单独指定数据
 *
 * read() 每次读取一个字符  返回值为读取内容的ASCII码 或者 Unicode编码 十进制 如果读取到末尾 返回-1
 * read(char [] data) 每次读取一个字符数组 返回值为读取字符的个数 如果读取到末尾 返回-1
 * close() 关闭资源
 */
public class TestFileReader1 {
    public static void main(String[] args) {
        // 定义FileReader对象
        FileReader fileReader = null;
        try {
            // 创建对象 并且指定读取文件
            fileReader = new FileReader("A.txt");

            // 调用read()方法 读取一个字符   接收返回值
            int readData = fileReader.read();

            // 打印读取数据 返回值为int类型 需要强转为char类型
            System.out.println("readData = " + (char)readData);

            readData = fileReader.read();

            System.out.println("readData = " + (char)readData);

            // 循环读取 循环依据为读取返回值不为-1 表示还未读取结束
            while((readData = fileReader.read()) != -1){
                System.out.println("readData = " + (char)readData);
            }

        } catch (FileNotFoundException e) {
            throw new RuntimeException(e);
        } catch (IOException e) {
            throw new RuntimeException(e);
        }finally{
            // 关闭资源
            try{
                // 在关闭之前做非空判断 因为如果为空 调用close方法 将出现空指针异常
                if(fileReader != null){
                    fileReader.close();
                }
            }catch(IOException e){
                e.printStackTrace();
            }
        }
    }
}

read(char [] data) 每次读取一个字符数组 返回值为读取字符的个数 如果读取到末尾 返回-1

package com.atguigu.test1;

import java.io.FileReader;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/21 9:20
 * read(char [] data) 每次读取一个字符数组 返回值为读取字符的个数 如果读取到末尾 返回-1
 */
public class TestFileReader2 {
    public static void main(String[] args) throws IOException {
        // 创建对象 并且指定读取文件
        FileReader reader = new FileReader("A.txt");

        // 创建长度为10的char数组 用于保存读取数据
        char [] data = new char[10];

        // 定义每次读取字符的个数 初始化值为-1
        int readCount = -1;

        // 循环读取 循环条件为 读取字符个数不为-1
        while((readCount = reader.read(data)) != -1){
            // 将读取完毕的字符数组 转换为字符串
            // 第一个参数 为字符数组
            // 第二个参数 转换为字符串的起始位置
            // 第三个参数 转换个数
            System.out.println(new String(data,0,readCount));
        }

        reader.close(); // 关闭资源

    }
}

BufferedReader

Reader

BufferedReader 带有缓冲区的字符读取流 也叫缓冲流 可以提高读取文件的效率

因为读取过程是先将内容 读取到内容 等待一行读取完毕 再一次行加载到程序中 相当于减少了

与硬盘 内存 IO的次数 所以效率更高

String readLine() 每次读取一行内容

InputStreamReader 可以作为参数构造BufferedReader实例

而InputStreamReader支持使用FileInputStream 字节读取流构建实例

所以 InputStreamReader属于转换流

package com.atguigu.test2;

import java.io.*;

/**
 * Reader
 * BufferedReader  带有缓冲区的字符读取流 也叫缓冲流 可以提高读取文件的效率
 * 因为读取过程是先将内容 读取到内容 等待一行读取完毕 再一次行加载到程序中 相当于减少了
 * 与硬盘 内存 IO的次数 所以效率更高
 * <p>
 * String readLine() 每次读取一行内容
 * <p>
 * InputStreamReader 可以作为参数构造BufferedReader实例
 * 而InputStreamReader支持使用FileInputStream 字节读取流构建实例
 * 所以 InputStreamReader属于转换流
 */
public class TestBufferedReader1 {
    public static void main(String[] args) {
        // 声明字节流对象
        FileInputStream fileInputStream = null;
        // 声明字符流/转换流 对象
        InputStreamReader inputStreamReader = null;
        // 声明缓冲流对象
        BufferedReader reader = null;
        try {
            // 指定读取文件
            fileInputStream = new FileInputStream("A.txt");

            // 创建转换流对象 需要使用字节流对象作为参数
            inputStreamReader = new InputStreamReader(fileInputStream);

            // 创建缓冲流对象 使用转换流作为参数
            reader = new BufferedReader(inputStreamReader);

            // 声明读取内容返回值 因为每次读取一行 所以返回值为String类型
            String line = null;

            // 循环读取 循环依据为返回值不为空 表示还未读取完毕
            while ((line = reader.readLine()) != null) {
                System.out.println(line); // 打印读取内容
            }

        } catch (FileNotFoundException e) {
            throw new RuntimeException(e);
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 关闭资源 先用后关
            // 关闭之前做非空判断
            try {
                if (reader != null) {
                    reader.close();
                }
                if (inputStreamReader != null) {
                    inputStreamReader.close();
                }
                if (fileInputStream != null) {
                    fileInputStream.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }


        }

    }


}

Writer
OutputStreamWriter

Writer 字符写入流 父类 抽象类

OutputStreamWriter 写入的转换流

OutputStreamWriter(OutputStream )

OutputStreamWriter(OutputStream outputStream,String charSetName)

write(String str) : 写入内容 直接写入字符串

flush() 刷新内容 将内存中的内容刷新到硬盘上

close() 关闭资源

package com.atguigu.test3;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;

/**
 *  Writer 字符写入流 父类 抽象类
 *  OutputStreamWriter 写入的转换流
 *  OutputStreamWriter(OutputStream )
 *  OutputStreamWriter(OutputStream outputStream,String charSetName)
 *
 *  write(String str) : 写入内容 直接写入字符串
 *  flush() 刷新内容 将内存中的内容刷新到硬盘上
 *  close() 关闭资源
 *
 */
public class TestOutputStreamWriter1 {
    public static void main(String[] args) {
        // 声明字节写入流对象
        FileOutputStream fileOutputStream = null;
        // 声明字符写入流/转换流 对象
        OutputStreamWriter writer = null;
        try {
            // 创建字节写入流对象
            // 第一个参数 表示写入当前相对路径下(相对于当前项目而言的路径) B.txt 如果文件不存在 将自动创建
            // 第二个参数 true 表示在原文件基础之上追加内容 false或者不写 表示覆盖原内容
            fileOutputStream = new FileOutputStream("B.txt",true);

            // 创建字符写入流 / 转换流对象  使用字节流对象作为参数构造实例
            writer =  new OutputStreamWriter(fileOutputStream);

            // 写入文件
            writer.write("123456");

            writer.flush(); // 刷新 表示将缓存中的数据刷新到硬盘中

        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }finally{
            // 关闭资源 关闭资源之前 会自动刷新缓存
            try {
                if(writer != null){
                    writer.close();
                }
                if(fileOutputStream != null){
                    fileOutputStream.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }


    }
}

OutputStreamWriter(OutputStream outputStream,String charSetName) 指定写入文件的编码格式

package com.atguigu.test3;

import jdk.internal.util.xml.impl.Input;

import java.io.*;
import java.nio.Buffer;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/21 11:32
 *  OutputStreamWriter(OutputStream outputStream,String charSetName) 指定写入文件的编码格式
 */
public class TestOutputStreamWriter2 {
    public static void main(String[] args) throws IOException {

        // 打印本地平台默认的编码格式
        System.out.println(System.getProperty("file.encoding"));

        // 创建字节流对象 如果文件不存在将自动创建
        FileOutputStream fileOutputStream = new FileOutputStream("C.txt");

        // 创建字符流/转换流 对象  指定写入文件编码格式为 'GBK'
        OutputStreamWriter writer = new OutputStreamWriter(fileOutputStream,"GBK");

        // 写入文件
        writer.write("hello world 世界你好 66666");

        // 刷新缓存
        writer.flush();
        // 关闭资源
        writer.close();

        FileInputStream fileInputStream = new FileInputStream("C.txt");

        // 因为写入的编码格式为 GBK 所以读取的编码格式也应该设置为  GBK 否则将出现乱码
        InputStreamReader inputStreamReader = new InputStreamReader(fileInputStream,"GBK");

        BufferedReader reader = new BufferedReader(inputStreamReader);

        System.out.println(reader.readLine());

        reader.close();

        inputStreamReader.close();

        fileOutputStream.close();


    }
}

FileWriter

Writer

OutputStreamWriter

FileWriter : 只能按照本地平台默认的编码格式写入文件

package com.atguigu.test4;

import java.io.FileWriter;
import java.io.IOException;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/21 14:01
 *  Writer
 *  OutputStreamWriter
 *  FileWriter : 只能按照本地平台默认的编码格式写入文件
 */
public class TestFileWriter {
    public static void main(String[] args) {
        // 声明写入文件对象
        FileWriter writer = null;
        try {
            // 创建对象 并且指定写入文件  如果文件不存在 将会自动创建
            // 默认不写第二个参数 表示覆盖原文件内容
            writer = new FileWriter("D.txt");

            // 写入内容
            writer.write("hello world 世界你好 6666");

            writer.flush(); // 刷新缓存
        } catch (IOException e) {
            e.printStackTrace();
        }finally{
            // 关闭资源
            try {
                if(writer != null){
                    writer.close();
                }
            } catch(IOException e) {
                e.printStackTrace();
            }
        }
    }
}
BufferedWriter

BufferedWriter : 带有缓存区的字符写入流 写入的内容先保存在缓冲区 调用flush方法 将缓存区中的内容

刷新到硬盘上 提高写入文件的效率

独有 newLine() 换行

package com.atguigu.test4;

import java.io.*;

/**
 * @author WHD
 * @description TODO
 * @date 2023/8/21 14:08
 *  BufferedWriter : 带有缓存区的字符写入流  写入的内容先保存在缓冲区 调用flush方法 将缓存区中的内容
 *  刷新到硬盘上 提高写入文件的效率
 *  独有 newLine() 换行
 */
public class TestBufferedWriter {
    public static void main(String[] args) throws IOException {
        // 创建对象 以匿名对象的方式作为参数构造实例
        BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(new FileOutputStream("E.txt")));

        // 写入文件 \n 和 newLine() 方法 都可以实现换行的效果
        writer.write("hello world \n 世界你好 \n 6666   1");
        writer.newLine();
        writer.write("hello world \n 世界你好 6666  \n  2");
        writer.flush(); // 刷新缓存
        writer.close(); // 关闭资源

    }
}

三、常用场景与代码示例

1. ​文件复制(字节流 + 缓冲)​
try (InputStream is = new FileInputStream("source.jpg");
     BufferedInputStream bis = new BufferedInputStream(is);
     OutputStream os = new FileOutputStream("dest.jpg");
     BufferedOutputStream bos = new BufferedOutputStream(os)) {

    byte[] buffer = new byte[1024];
    int len;
    while ((len = bis.read(buffer)) != -1) {
        bos.write(buffer, 0, len);
    }
} catch (IOException e) {
    e.printStackTrace();
}
2. ​读取文本文件(字符流 + 缓冲)​
try (Reader reader = new FileReader("text.txt", StandardCharsets.UTF_8);
     BufferedReader br = new BufferedReader(reader)) {

    String line;
    while ((line = br.readLine()) != null) {
        System.out.println(line);
    }
} catch (IOException e) {
    e.printStackTrace();
}
3. ​对象序列化
// 序列化对象到文件
try (ObjectOutputStream oos = new ObjectOutputStream(
        new FileOutputStream("user.dat"))) {
    User user = new User("Alice", 30);
    oos.writeObject(user);
}

// 反序列化
try (ObjectInputStream ois = new ObjectInputStream(
        new FileInputStream("user.dat"))) {
    User user = (User) ois.readObject();
    System.out.println(user.getName()); // 输出:Alice
}

四、关键注意事项

  1. 资源释放

    • 使用 ​**try-with-resources** 语法自动关闭流(实现 AutoCloseable 接口)。
    • 手动关闭时需按 ​先开后关 的顺序。
  2. 字符编码问题

    • 明确指定字符编码(如 UTF-8),避免默认编码导致乱码:
      new InputStreamReader(new FileInputStream("file.txt"), StandardCharsets.UTF_8);
  3. 性能优化

    • 使用 ​缓冲流​(如 BufferedInputStreamBufferedReader)减少 I/O 次数。
    • 处理大文件时,避免一次性读取全部内容到内存。
  4. NIO 补充

    • ​**Files 工具类**​(简化文件操作):
      Files.copy(Path.of("source.txt"), Path.of("dest.txt"));
      List<String> lines = Files.readAllLines(Path.of("text.txt"));
    • 通道(Channel)与缓冲区(Buffer)​:适用于高性能 I/O(如网络通信)。

五、总结对比

流类型 适用场景 优点 缺点
字节流 二进制文件(图片、视频) 通用性强,处理任意数据 文本处理需手动编解码
字符流 文本文件 自动处理字符编码,简化文本操作 不适用于二进制数据
缓冲流 高频 I/O 操作 显著提升性能 增加代码层级
对象流 对象序列化 直接读写对象 需实现 Serializable

掌握 Java I/O 流是处理数据持久化、网络通信的基础,理解其分类与设计模式(如装饰器模式)能更灵活应对复杂场景。