一文详解如何使用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流实现文件数据读写及文件复制内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Spring学习通过AspectJ注解方式实现AOP操作

    Spring学习通过AspectJ注解方式实现AOP操作

    这篇文章主要为大家介绍了Spring学习通过AspectJ注解方式实现AOP操作,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-05-05
  • 关于SpringBoot中的跨域问题

    关于SpringBoot中的跨域问题

    这篇文章主要介绍了关于SpringBoot中的跨域问题,同源策略是由Netscape提出的一个著名的安全策略,它是浏览器最核心也最基本的安全功能,现在所有支持JavaScript的浏览器都会使用这个策略,需要的朋友可以参考下
    2023-08-08
  • selenium+java破解极验滑动验证码的示例代码

    selenium+java破解极验滑动验证码的示例代码

    本篇文章主要介绍了selenium+java破解极验滑动验证码的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-01-01
  • Java 中的 getDeclaredFields()使用与原理解析

    Java 中的 getDeclaredFields()使用与原理解析

    在Java反射机制中,getDeclaredFields()用于获取类的所有字段,包括私有字段,通过反射,可以在运行时动态地获取类的信息并操作其成员,本文详细介绍了getDeclaredFields()的使用方法、工作原理以及最佳实践,涵盖了反射的基本概念、使用场景和注意事项,感兴趣的朋友一起看看吧
    2025-01-01
  • Java多线程之多线程异常捕捉

    Java多线程之多线程异常捕捉

    在java多线程程序中,所有线程都不允许抛出未捕获的checked exception,也就是说各个线程需要自己把自己的checked exception处理掉,通过此篇文章给大家分享Java多线程之多线程异常捕捉,需要的朋友可以参考下
    2015-08-08
  • Elasticsearch Analyzer 内置分词器使用示例详解

    Elasticsearch Analyzer 内置分词器使用示例详解

    这篇文章主要为大家介绍了Elasticsearch Analyzer 内置分词器使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-11
  • 一起来看看springboot集成redis的使用注解

    一起来看看springboot集成redis的使用注解

    这篇文章主要为大家详细介绍了springboot集成redis的使用注解,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03
  • 浅谈Java设计模式系列-装饰器模式

    浅谈Java设计模式系列-装饰器模式

    这篇文章主要介绍了Java设计模式系列-装饰器模式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • Java使用Poi导出Excel表格方法实例

    Java使用Poi导出Excel表格方法实例

    这篇文章主要给大家介绍了关于Java使用Poi导出Excel表格的相关资料,Java POI是一个用于操作Microsoft Office格式的Java API库,可以使用它来导出Excel文件,需要的朋友可以参考下
    2023-10-10
  • Java实现最小生成树MST的两种解法

    Java实现最小生成树MST的两种解法

    最小生成树(MST)指在连通图的所有生成树中,所有边的权值和最小的生成树。本文介绍了求最小生成树的两种方法:Prim算法和Kruskal算法,需要的可以参考一下
    2022-05-05

最新评论