一文详解如何使用IO流实现文件数据的读写及文件复制

 更新时间:2025年10月09日 11:45:10   作者:68岁敲代码老婆婆  
Java中的IO流是基础操作之一,包括控制台输入、控制台输出、读写文件等,下面这篇文章主要介绍了如何使用IO流实现文件数据的读写及文件复制的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下

一. 什么是IO流?

IO流本质上是 Java 中用于处理设备间数据传输的 API(不止限于文件,还包括网络、内存、键盘等),可以实现对文件数据的读写,区别于File类只能操作文件本身

  • 输入流(Input):数据从外部设备(如文件、网络)进入程序(内存);
  • 输出流(Output):数据从程序(内存)发送到外部设备。File 类仅能操作文件的 “元数据”(如创建、删除、判断存在性),而 IO 流负责文件内容的读写,这是两者的核心区别。

二. IO流的分类

1. 字节IO流:

以字节(8bit)为单位读写数据,可处理所有类型文件(文本、图片、视频等)

输入:FileInputStream()     对应的缓冲流: BufferedInputStream() 

 输出:FileOutputStream()   对应的缓冲流: BufferedOutputStream() 

2.字符IO流:

以字符(16bit,Java 中char)为单位读写数据,仅适合处理文本文件

输入:FileReader()             对应的缓冲流: BufferedReader() 

输出:FileWriter()                对应的缓冲流: BufferedWriter() 

注意:为什么要记对应的缓冲流?

缓冲流(BufferedXXX)通过内置8KB 缓冲区(字节缓冲流)或字符缓冲区(字符缓冲流),减少直接与磁盘的 IO 次数(磁盘 IO 效率远低于内存操作),从而提升性能。

  • 例如:读取文件时,缓冲流会一次性从磁盘读取 8KB 数据到缓冲区,程序从缓冲区获取数据;写入时先攒满缓冲区再一次性写入磁盘。

使用规范

处理流需 “包裹” 节点流,关闭时只需关闭外层处理流(会自动关闭内层节点流):

// 示例:缓冲流包裹节点流
try (BufferedReader br = new BufferedReader(new FileReader("test.txt"))) {
    String line;
    while ((line = br.readLine()) != null) {
        // 读取数据
    }
} catch (IOException e) {
    e.printStackTrace();
}

其他重要补充

  • 流的继承关系

    • 所有字节输入流继承InputStream,字节输出流继承OutputStream
    • 所有字符输入流继承Reader,字符输出流继承Writer(这四个是抽象基类,不能直接实例化)。
  • 编码问题:字符流涉及编码转换,FileReader/FileWriter默认使用系统编码(可能导致乱码),建议用InputStreamReader/OutputStreamWriter指定编码:

    // 指定UTF-8编码读取文本
    Reader reader = new InputStreamReader(new FileInputStream("test.txt"), "UTF-8");
    
  • JDK7 + 的 try-with-resources:流实现了AutoCloseable接口,可在 try 后自动关闭,无需手动调用close(),推荐优先使用(如上述缓冲流示例)。

思维导图

练习一: 统计单一文件目录大小

package com.itheima.homework;

import java.io.File;

/**
 * @author Administrator
 **需求:**
    假设在`D:\itheima\`目录下有若干个文件**(只有文件没有目录)**,请编写程序统计`D:\itheima`目录的大小。
 **提示:**
1. 如果没有`D:\itheima`目录,就在任意盘下创建一个`itheima`目录
2. 统计目录的大小就是统计目录中所有文件的大小之和
 */
public class Work1 {
    public static void main(String[] args) {
        File file = new File("C:\\Users\\Administrator\\IDEA\\java-upgrade\\day06-file-recursion-io\\src\\com\\itheima\\d4_io\\d1_byteio");
        File[] files = file.listFiles();
        long length = 0;
        for (File file1 : files) {
            length+=file1.length();
        }
        System.out.println("总大小为"+length);

    }


}

练习2:统计目录若干文件大小

package com.itheima.homework;

import java.io.File;

/**
 * @author Administrator
 **需求:
        假设在`D:\itheima\`目录下有若干个文件和目录,请编写程序统计`D:\itheima`目录的大小。

 **提示:**统计目录的大小就是统计目录(及其子目录)中所有文件的大小之和
 */
public class Work2 {
    public static void main(String[] args) {
        File file = new File("day06-file-recursion-io");
        System.out.println("总大小为"+fileSizeCount(file));
    }
    public static long fileSizeCount(File dir){

        File[] files = dir.listFiles();
        long length = 0;
        for (File file1 : files) {
            if (file1.isFile()) {
                long len = file1.length();
                System.out.println(file1.getName()+"-->"+len);
                length+=file1.length();
            }else{
//                return fileSizeCount(file1); 不能return直接否定了外层循环
//                fileSizeCount(file1);
                //todo 没有累加非同级目录的文件大小,结果为909,只计算了1级目录的文件大小
                length +=fileSizeCount(file1);
                //递归就应该累加文件大小,就跟阶乘类似
                System.out.println("子级目录长度:"+ length);


            }
        }
    return length;
    }

}

练习3: 复制单个文件到目录中

package com.itheima.homework;

import java.io.*;

/**
 * @author Administrator
 * **需求:**
 假设在`D:\itheima\`目录下有若干个文件**(只有文件没有目录)**,请编写程序将`D:\itheima`目录中的**一个文件**复制到当前模块下的`itheima`目录中,文件名不变。
 * **要求:**不使用commons-io框架
 */
public class Work3 {
    public static void main(String[] args) throws IOException {
        copyFile();
    }
    public static void copyFile()  {
        //复制文件的思路:字节流符合视频\图片\音频\文本,更普遍使用
        try(
                FileInputStream fis = new FileInputStream(new File("day06-file-recursion-io\\src\\com\\itheima\\d1_file\\Demo1.java"));
                FileOutputStream fos = new FileOutputStream("day06-file-recursion-io\\src\\com\\itheima\\homework\\Demo1.java");

        ) {
                       byte[] bytes = new byte[1024];
            int lenth = 0;
            while((lenth=fis.read(bytes))!=-1){
                fos.write(bytes,0,lenth);
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
}

练习4:将某一目录下所有文件复制到别的目录下

package com.itheima.homework;

import java.io.*;

/**
 * @author Administrator
 * 假设在`D:\itheima\`目录下有若干个文件**(只有文件没有目录)**,请编写程序将`D:\itheima`目录中的**所有文件**复制到当前模块下的`itheima`目录中。
 */
/*todo Error: FileInputStream 只能用于读取文件,不能读取目录
*   遍历目录中的每个文件
    为每个源文件创建独立的 FileInputStream
    为目标文件创建 FileOutputStream
    *
    * 输出目录拒绝访问的原因是:
    FileOutputStream 不能直接写入目录
    FileOutputStream 只能用于创建或写入文件,不能直接写入目录
    代码中 new FileOutputStream(outputFile) 试图将数据写入目录 work4,这是不允许的
    目标路径应该是文件而不是目录
    每个源文件都需要对应一个目标文件
    当前代码试图将所有文件内容都写入同一个目录路径*
    *使用 while((len = fis.read(bytes)) != -1) 循环读取
    每次读取后立即写入目标文件
    直到 read() 返回 -1(文件结束)才停止
    这样就能完整复制整个文件内容,而不是只复制前1024字节。*/
public class Work4 {
    public static void main(String[] args) throws IOException {
        File inputFile = new File("day06-file-recursion-io\\src\\com\\itheima\\d4_io\\d1_byteio");
        File outputFile = new File("day06-file-recursion-io\\src\\com\\itheima\\homework\\work4");
        copyFiles(inputFile,outputFile);
    }
    public static void copyFiles(File  dir,File outputFile) throws IOException {
        //创建字节输入输出流
        //这里的都是目录

            //读取输入流目录的子级
            File[] files = dir.listFiles();
            for (File file : files) {
                //遍历父级目录下的文件是否是文件
                if (file.isFile()) {
                    try ( FileInputStream fis = new FileInputStream(file);
                          //todo 此时字节输入流已经拿到了文件
                          FileOutputStream fos = new FileOutputStream(new File(outputFile, file.getName()));
                          //todo 为输出字节流的目标文件指定具体文件名,父目录+子文件名
                          ){
                        byte[] bytes = new byte[1024];
                        int len = 0;
                        while ((len=fis.read(bytes))!=-1){
                             fos.write(bytes,0,len);
                         }
                    } catch (IOException e) {
                        throw new RuntimeException(e);
                    }
                }else{
                    copyFiles(file, new File(outputFile, file.getName()));
                }
            }
            //读入和输出
            //关闭流

    }
}

练习5:符缓冲流读取数据并封装对象

package com.itheima.homework;

import java.io.*;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

/**
 * @author Administrator
 * - 使用字符缓冲流读取”students.txt”文件,将每行数据封装为一个Student对象,并将Student对象存储到一个集合
 *
 * - 遍历并打印集合的所有Student信息
 */
public class Work5 {
    public static void main(String[] args) throws IOException {
        Reader file = new FileReader("day06-file-recursion-io\\student.txt");
        BufferedReader br = new BufferedReader(file);
        //读取行
        String line = null;
        List<Student> studentArrayList = new ArrayList<>();
        while((line=br.readLine())!=null){
            String s = new String(line);
            String[] split = s.split(",");
//            System.out.println(Arrays.toString( split));
            Student student = new Student(split[0], Integer.parseInt(split[1]));

            studentArrayList.add(student);

            /*//todo 而不是“每行都重复 3 次”,根本原因是:
                    //你把 new ArrayList<>() 写在了 while 循环里
                    //→ 每读一行就 new 一个新的空集合 → 只装当前这一个 Student → 打印完就弃用 → 下一行再 new 一个新的空集合……*/
        }
        br.close();
        System.out.println(studentArrayList);


    }
}

总结 

到此这篇关于如何使用IO流实现文件数据的读写及文件复制的文章就介绍到这了,更多相关IO流实现文件数据读写及文件复制内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • JAVA中对List进行查询

    JAVA中对List进行查询

    下面小编就为大家带来一篇在java List中进行模糊查询的实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2021-08-08
  • 使用Maven进行依赖排除的详细步骤

    使用Maven进行依赖排除的详细步骤

    在Maven中,依赖排除是一种常见的技术,用于从项目的依赖中排除特定的传递性依赖,这通常用于解决依赖冲突或避免引入不需要的库,以下是如何在Maven中使用依赖排除的详细步骤,包括代码示例,需要的朋友可以参考下
    2024-11-11
  • Java实现猜数程序

    Java实现猜数程序

    这篇文章主要为大家详细介绍了Java实现猜数程序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-12-12
  • Java 实现协同过滤算法推荐算法的示例代码

    Java 实现协同过滤算法推荐算法的示例代码

    本文介绍了协同过滤算法的概念,包括基于用户的协同过滤和基于物品的协同过滤,文章详细解释了数据准备、相似度计算以及如何在Java中实现这些算法,通过一个简单的用户-物品评分矩阵示例,展示了如何计算用户和物品之间的相似度,并推荐未评分的物品,感兴趣的朋友一起看看吧
    2025-02-02
  • Java对象布局(JOL)实现过程解析

    Java对象布局(JOL)实现过程解析

    这篇文章主要介绍了Java对象布局(JOL)实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • Java ResourceBundle案例详解

    Java ResourceBundle案例详解

    这篇文章主要介绍了Java ResourceBundle案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • Java多线程--让主线程等待所有子线程执行完毕在执行

    Java多线程--让主线程等待所有子线程执行完毕在执行

    Java主线程等待所有子线程执行完毕在执行,其实在我们的工作中经常的用到,本篇文章就介绍了Java多线程--让主线程等待所有子线程执行完毕在执行,有需要的可以了解一下。
    2016-11-11
  • Java Servlet实现表白墙的代码实例

    Java Servlet实现表白墙的代码实例

    最近用Servlet做了个小项目,分享给大家,下面这篇文章主要给大家介绍了关于Java Servlet实现表白墙的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-02-02
  • 简单了解springboot加载配置文件顺序

    简单了解springboot加载配置文件顺序

    这篇文章主要介绍了简单了解springboot加载配置文件顺序,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-06-06
  • 一文详解各种ElasticSearch查询在Java中的实现

    一文详解各种ElasticSearch查询在Java中的实现

    Elasticsearch是用Java开发的,并作为Apache许可条款下的开放源码发布,是当前流行的企业级搜索引擎,下面这篇文章主要给大家介绍了关于各种ElasticSearch查询在Java中实现的相关资料,需要的朋友可以参考下
    2023-11-11

最新评论