JavaWeb~使用字符流与字节流 分别 实现二进制文件与文本文件的复制

流Stream是什么?

  • 在 Java中所有数据都是使用流读写的。流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象。即数据在两设备间的传输称为流,流的本质是数据传输,根据数据传输特性将流抽象为各种类,方便更直观的进行数据操作。
  • 按照流向分:输入流;输出流 (单纯站在数据角度: CPU直接从内存里读取数据,不会和外设打交道,为什么呢?因为硬盘太慢了, 所以数据要能够被CPU处理,就必须先被拿到内存,才能够被CPU处理)
  • 输入就是将数据从各种输入设备(包括文件、键盘等)中读取到内存中。
  • 输出则正好相反,是将数据写入到各种输出设备(比如文件、显示器、磁盘等)。
  • 例如键盘就是一个标准的输入设备,而显示器就是一个标准的输出设备,但是文件既可以作为输入设备,又可以作为输出设备。
  • 按照处理数据的单位分:字节流(8位的字节);字符流(16位的字节)
  1. 字节流操作的基本单元是字节;字符流操作的基本单元为Unicode码元。
  2. 字节流在操作的时候本身不会用到缓冲区的,是与文件本身直接操作的;而字符流在操作的时候使用到缓冲区的。
  3. 所有文件的存储都是字节(byte)的存储,在磁盘上保留的是字节。
  4. 在使用字节流操作中,即使没有关闭资源(close方法),也能输出;而字符流不使用close方法的话,不会输出任何内容。
  • 流对象的核心操作就是
  1. 打开文件(构造函数)
  2. read 把文件的数据读到内存中
  3. write 把数据从内存中写入文件中
  4. close 关闭文件 (如果不关闭文件就会造成文件泄漏 所谓文件泄漏就是一个PCB中会有一个数组一样的东西去记录打开的文件 这个数组的大小是有限的如果不关闭一直打开文件就会让数组满了 后序的文件就会打开失败 这就是文件泄漏)

字节流实现复制图片(二进制文件)

  • FileInputStream(File file)
    通过打开与实际文件的连接创建一个 FileInputStream ,该文件由文件系统中的 File 对象 file 命名
  • FileInputStream(String name)
    通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的路径名 name 命名。
  • FileOutputStream(File file)
    创建文件输出流以写入由指定的 File 对象表示的文件。
  • FileOutputStream(String name)
    创建文件输出流以指定的名称写入文件。
import com.sun.media.sound.FFT;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

public class IoTest1 {
    public static void main(String[] args) {
        copyFile();
    }

    private static void copyFile() {

        try (FileInputStream fileInputStream = new FileInputStream("D:\\java\\test\\more\\flower.png");
             FileOutputStream fileOutputStream = new FileOutputStream("D:\\java\\test\\more\\flower2.png")){

            byte[] buffer = new byte[1024];
            int len = -1;
            while ((len = fileInputStream.read(buffer)) != -1) {
                fileOutputStream.write(buffer, 0, len);
            }
        } catch (IOException e) {
            e.getStackTrace();
        }

        System.out.println("复制成功");
    }
}

在这里插入图片描述
在这里插入图片描述

字节缓冲流实现复制图片(二进制文件)

  • 为什么需要有缓冲流?
    当我们用read()读取文件时,每读一个字节,访问一次硬盘,效率很低 。文件过大时,操作起来也不是很方便。因此我们需要用到buffer缓存流,当创建buffer对象时,会创建一个缓冲区数组。当我们读一个文件时,先从硬盘中读到缓冲区,然后直接从缓冲区输出即可,效率会更高。

  • BufferedInputStream(InputStream in)
    创建一个 BufferedInputStream 并保存其参数,输入流 in ,供以后使用。

  • BufferedOutputStream(OutputStream out)
    创建一个新的缓冲输出流,以将数据写入指定的底层输出流。

  • 使用BufferedOutputStream还有一个特殊操作就是 flush操作 (手动刷新缓冲区)就是把数据从缓冲区写入磁盘的IO设备

  • 缓冲区合适会被刷新

  1. 缓冲区满了的时候就会自动刷新
  2. 调用close方法的时候也会触发刷新
  3. 主动调用flush也会刷新
import java.io.*;

public class OiTest2 {
    public static void main(String[] args) {
        copyFile();
    }

    private static void copyFile() {
        try (BufferedInputStream bufferedInputStream
                     = new BufferedInputStream(new FileInputStream("D:\\java\\test\\more\\flower.png"));
             BufferedOutputStream bufferedOutputStream
                     = new BufferedOutputStream(new FileOutputStream("D:\\java\\test\\more\\flower2.png"))){


            byte[] buffer = new byte[1024];
            int len = -1;
            while ((len = bufferedInputStream.read(buffer)) != -1) {
                bufferedOutputStream.write(buffer, 0, len);
            }

        } catch (IOException e) {
            e.getStackTrace();
        }

        System.out.println("复制成功");
    }
}

  • 对比有无缓冲区的效率


import java.io.*;

public class IoTest1 {
    public static void main(String[] args) {
        copyFile();
        copyFileBuffer();
    }

    private static void copyFile() {
        long beg = System.currentTimeMillis();
        try (FileInputStream fileInputStream = new FileInputStream("D:\\java\\test\\more\\flower.png");
             FileOutputStream fileOutputStream = new FileOutputStream("D:\\java\\test\\more\\flower2.png")){

            byte[] buffer = new byte[1];
            int len = -1;
            while ((len = fileInputStream.read(buffer)) != -1) {
                fileOutputStream.write(buffer, 0, len);
            }
        } catch (IOException e) {
            e.getStackTrace();
        }

        System.out.println("复制成功");
        long end = System.currentTimeMillis();
        System.out.println("无buffer的时间为" + (end - beg) + "ms");
    }

    private static void copyFileBuffer() {
        long beg = System.currentTimeMillis();
        try (BufferedInputStream bufferedInputStream
                     = new BufferedInputStream(new FileInputStream("D:\\java\\test\\more\\flower.png"));
             BufferedOutputStream bufferedOutputStream
                     = new BufferedOutputStream(new FileOutputStream("D:\\java\\test\\more\\flower3.png"))){


            byte[] buffer = new byte[1];
            int len = -1;
            while ((len = bufferedInputStream.read(buffer)) != -1) {
                bufferedOutputStream.write(buffer, 0, len);
            }

        } catch (IOException e) {
            e.getStackTrace();
        }

        System.out.println("复制成功");
        long end = System.currentTimeMillis();
        System.out.println("有buffer的时间为" + (end - beg) + "ms");
    }
}

在这里插入图片描述

字符流实现复制文本

  • FileReader(File file)
    创建一个新的 FileReader ,给出 File 读取。
  • FileReader(String fileName)
    创建一个新的 FileReader ,给定要读取的文件的名称。
  • FileWriter(File file) 给一个File
    对象构造一个FileWriter对象。
  • FileWriter(String fileName)
    构造一个给定文件名的FileWriter对象。
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

public class IOTest3 {
    public static void main(String[] args) {
        copyFile();
    }

    private static void copyFile() {

        try (FileReader fileReader = new FileReader("D:\\java\\test\\more\\NBA.txt");
             FileWriter fileWriter = new FileWriter("D:\\java\\test\\more\\NBA2.txt")){

            char[] buffer = new char[1024];
            int len = -1;
            while ((len = fileReader.read(buffer)) != -1) {
                fileWriter.write(buffer, 0, len);

            }

        } catch (IOException e) {
            e.getStackTrace();
        }
    }
}

在这里插入图片描述

字符缓冲流实现复制文本

  • 为了提高字符流读写的效率,引入了缓冲机制,进行字符批量的读写,提高了单个字符读写的效率。BufferedReader 用于加快读取字符的速度,BufferedWriter 用于加快写入的速度。BufferedReader 和 BufferedWriter 类各拥有 8192个 字符的缓冲区。当 BufferedReader在 读取文本文件时,会先尽量从文件中读入字符数据并放满缓冲区,而之后若使用read()方法,会先从缓冲区中
    进行读取。如果缓冲区数据不足,才会再从文件中读取,使用 BufferedWriter时,写入的数据并不会先输出到目的地,而是先存储至缓冲区中。如果缓冲区中的数据满了,才会一次对目的地进行写出。
import java.io.*;

public class IOTest4 {
    public static void main(String[] args) {
        copyFile();
    }

    private static void copyFile() {

        try (BufferedReader bufferedReader = new BufferedReader(new FileReader("D:\\java\\test\\more\\NBA.txt"));
             BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter("D:\\java\\test\\more\\NBA2.txt"))){

            char[] buffer = new char[1024];
            int len = -1;
            while ((len = bufferedReader.read(buffer)) != -1) {
                bufferedWriter.write(buffer, 0, len);
            }

        } catch (IOException e) {
            e.getStackTrace();
        }

        System.out.println("复制成功");
    }
}

  • 还有一种一次获取一行的写法
import java.io.*;

public class IOTest4 {
    public static void main(String[] args) {
        copyFile();
    }

    private static void copyFile() {

        try (BufferedReader bufferedReader = new BufferedReader(new FileReader("D:\\java\\test\\more\\NBA.txt"));
             BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter("D:\\java\\test\\more\\NBA2.txt"))){

            String line;
            while ((line = bufferedReader.readLine()) != null) {
                bufferedWriter.write(line);
                //每次读入一行要换行 给line加 “\n" 或者下面这种都行
                bufferedWriter.newLine();
            }

        } catch (IOException e) {
            e.getStackTrace();
        }

        System.out.println("复制成功");
    }
}

  • 还有点小细节就是 进行文本输入输出的时候俩个文本编码要一致不然就会乱码
  • 在构造FileWriter FileOutPutStream 时 在构造函数里加上true就是衔接式读入 不加就是覆盖式读入 我上面演示的都是覆盖式读入
    在这里插入图片描述
    在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/Shangxingya/article/details/106850236