java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > IO流实现文件数据读写及文件复制

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

作者:68岁敲代码老婆婆

Java中的IO流是基础操作之一,包括控制台输入、控制台输出、读写文件等,下面这篇文章主要介绍了如何使用IO流实现文件数据的读写及文件复制的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下

一. 什么是IO流?

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

二. IO流的分类

1. 字节IO流:

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

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

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

2.字符IO流:

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

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

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

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

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

使用规范

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

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

其他重要补充

思维导图

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

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

您可能感兴趣的文章:
阅读全文