Java 你往我来的 IO 流,适合小白或傻傻分不清【输入、输出流】的伙计们 —— 第 03 篇 Java 的 字节输出缓冲流:BufferedOutputStream 字节输入缓冲流:Buff ..

Java 你往我来的 IO 流【第 03 篇】 Java 的 字节输出、输入缓冲流

字节输出缓冲流:BufferedOutputStream
字节输入缓冲流:BufferedInputStream

        关于 Java 常用的 IO 流,整理了 07 篇文章,这是第 03 篇。如果是小白,或者一直没弄清楚 IO 流,请依次把这几篇文章过一遍。当然,高手就不用来我的博客浪费青春了。

        小二,上代码:


import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;

/*
 * 这个类的知识点(字节缓冲流)
 * 1、使用 字节输出缓冲流 BufferedOutputStream 把数据写入文件
 * 2、使用 字节输入缓冲流 BufferedInputStream 把数据读到内存
 * 3、使用 字节输入缓冲流、输出缓冲流 将图片复制到指定路径
 */
public class ByteBufferedStream
{
    // 输出流:将数据从内存写入硬盘文件等,简称:写入输出外界(Writer、OutputStream)
    // BufferedOutputStream 和 FileOutputStream 区别:前者有缓冲区,后者没有
    // 输入流:将数据从硬盘文件等写入内存,简称:读取输入内存(Reader、InputStream)
    // BufferedInputStream 和 FileInputStream 区别:前者有缓冲区,后者没有
    // 缓冲区作用:在读取数据时,先把数据放到一个缓冲区里(内存),然后再批量进行处理,速度快。如果没有缓冲区,直接操作文件,效率低,性能差。因为读取内存比读取硬盘速度快得多
    // 流分两种:字节流和字符流
    // 字节流适用场景:主要用来处理字节或二进制对象。如:音频文件、图片、歌曲
    // 字符流适用场景:主要用来处理字符或字符串。如:关系到中文(文本)
    
    public static void main(String[] args)
    {
        File file = new File("D:" + File.separator + "temp" + File.separator + "test.txt"); // 在 Windows 下的路径分隔符和 Linux 下的路径分隔符是不一样的,如 Windows 的路径:D:\test.txt,而 Linux 则是:D:/test.txt,所以如果考虑跨平台,最好是用 File.separator 去实现分隔符,它能兼容二者
        File picture = new File("D:" + File.separator + "temp" + File.separator + "abc.jpg"); // 要被复制的源文件(图片)
        
        // 创建目录(这里如果不执行创建,那么必须保证 D 盘上有 temp 目录)
        //file.getParentFile().mkdirs();
        // 创建文件。这里不用手动创建也可以,因为下面在创建 FileOutputStream 实例时会自动创建
        //file.createNewFile();
        
        // 调用字节输出缓冲流方法(把内存数据写入文件)
        writeToFile(file);
        
        // 调用字节输入缓冲流方法(把文件数据读取到内存)
        readFromFile(file);
        
        // 调用字节输出缓冲流、输入缓冲流方法复制文件(图片)到指定路径
        copyFileByByteStream(picture);
    }
    
    // 字节输出缓冲流(把内存数据写入文件)
    public static void writeToFile(File file)
    {
        OutputStream os = null; // 为什么不放到 try 里面声明?如果把它放 try 里面声明,那么它的作用域仅限于 try,即只能在 try 里面使用它
        BufferedOutputStream obs = null; // 同上
        try
        {
            // FileOutputStream 是 OutputStream(抽象类,不能实例化)的子类
            os = new FileOutputStream(file); // FileOutputStream 没有缓冲区,建议使用输出缓冲流 BufferedOutputStream
            obs = new BufferedOutputStream(os); // 字节输出缓冲流,需要 FileOutputStream 作为参数
            String str = "101_编程世界_Hello World\r\n102_编程世界_Hello World\r\n103_编程世界_Hello World";
            byte[] b = str.getBytes(); // str.getBytes() 主要是把字符串转换为字节,因为字节流写入数据时是用字节
            obs.write(b);
            // 字节缓冲流使用了缓冲区,将数据写入文件时,数据并没有直接写入文件,而是保存在缓冲区,缓冲区满了之后再写入文件。flush() 方法强制将数据立即写入文件并清空缓冲区
            // flush() 不是必须的,但一定要调用 close() 方法
            // 如果 flush() 和 close() 方法都不调用,那么数据不能写入文件
            obs.flush();
        } catch (FileNotFoundException e)
        {
            e.printStackTrace();
        } catch (IOException e)
        {
            e.printStackTrace();
        } finally // finally 作用:无论是否抛出异常,finally 代码块总是会被执行,这里用来释放资源
        {
            try
            {
                // 关闭流(如果不关闭流,那么相应的垃圾回收机制就不进行回收,导致垃圾越来越多,浪费资源)
                if(obs != null)
                {
                    obs.close();
                }
                if(os != null)
                {
                    os.close();
                }
            } catch (IOException e)
            {
                e.printStackTrace();
            }
        }
    }
    
    // 字节输入缓冲流(把文件数据读取到内存)
    public static void readFromFile(File file)
    {
        InputStream is = null; // 为什么不放到 try 里面声明?如果把它放 try 里面声明,那么它的作用域仅限于 try,即只能在 try 里面使用它
        BufferedInputStream bis = null; // 同上
        try
        {
            is = new FileInputStream(file); // FileInputStream 没有缓冲区,建议使用输入缓冲流 BufferedInputStream
            bis = new BufferedInputStream(is); // 字节输入缓冲流,需要 FileInputStream 作为参数
            byte[] b = new byte[1024];
            int len;
            // 一个文件可能很大,每次读取 1024 个字节数据并放到字节变量 b 里面,用 while 循环判断是否读完,如果没有读完,这里将继续读后面的数据,直到全部读完返回 -1 结束循环
            while((len = bis.read(b)) != -1) // 如果达到流的末尾返回 -1(把文件读完之后返回 -1)
            {
                System.out.println(new String(b,0,len)); // new String(b) 将字节转换为字符串。0 和 len 代表从第 0 个到第 len 个,len 是每次读取数据的长度
            }
        } catch (FileNotFoundException e)
        {
            e.printStackTrace();
        } catch (IOException e)
        {
            e.printStackTrace();
        } finally // finally 作用:无论是否抛出异常,finally 代码块总是会被执行,这里用来释放资源
        {
            try
            {
                // 关闭流(如果不关闭流,那么相应的垃圾回收机制就不进行回收,导致垃圾越来越多,浪费资源)
                if(bis != null)
                {
                    bis.close();
                }
                if(is != null)
                {
                    is.close();
                }
            } catch (IOException e)
            {
                e.printStackTrace();
            }
        }
    }
    
    // 字节输出缓冲流、输入缓冲流复制文件(图片)到指定路径
    public static void copyFileByByteStream(File file)
    {
        InputStream is = null; // 为什么不放到 try 里面声明?如果把它放 try 里面声明,那么它的作用域仅限于 try,即只能在 try 里面使用它
        OutputStream os = null; // 同上
        BufferedInputStream bis = null; // 同上
        BufferedOutputStream bos = null; // 同上
        try
        {
            // FileInputStream 是 InputStream(抽象类,不能实例化)的子类
            is = new FileInputStream(file); // FileInputStream 没有缓冲区,建议使用输入缓冲流 BufferedInputStream。这里的 file 是要被复制的源文件
            bis = new BufferedInputStream(is); // 字节输入缓冲流,需要 FileInputStream 作为参数
            
            // FileOutputStream 是 OutputStream(抽象类,不能实例化)的子类
            os = new FileOutputStream(new File("D:\\temp\\pic副本.jpg")); // 这里的 new File() 表示把文件复制到目的地及文件名称
            // FileOutputStream 没有缓冲区,建议使用输出缓冲流 BufferedOutputStream
            bos = new BufferedOutputStream(os);
            byte[] b = new byte[1024];
            int len;
            // 一个文件可能很大,每次读取 1024 个字节数据并放到字节变量 b 里面,用 while 循环判断是否读完,如果没有读完,这里将继续读后面的数据,直到全部读完返回 -1 结束循环
            while((len = bis.read(b)) != -1) // 如果达到流的末尾返回 -1(把文件读完之后返回 -1)
            {
                bos.write(b,0,len); // 将字节写入文件。0 代表从哪里开始写入,len 表示从哪里结束输入
            }
            // 字节缓冲流使用了缓冲区,将数据写入文件时,数据并没有直接写入文件,而是保存在缓冲区,缓冲区满了之后再写入文件。flush() 方法强制将数据立即写入文件并清空缓冲区
            // flush() 不是必须的,但一定要调用 close() 方法
            // 如果 flush() 和 close() 方法都不调用,那么数据不能写入文件
            bos.flush();
        } catch (FileNotFoundException e)
        {
            e.printStackTrace();
        } catch (IOException e)
        {
            e.printStackTrace();
        } finally // finally 作用:无论是否抛出异常,finally 代码块总是会被执行,这里用来释放资源
        {
            try
            {
                // 关闭流(如果不关闭流,那么相应的垃圾回收机制就不进行回收,导致垃圾越来越多,浪费资源)
                if(bos != null)
                {
                    bos.close();
                }
                if(os != null)
                {
                    os.close();
                }
                if(bis != null)
                {
                    bis.close();
                }
                if(is != null)
                {
                    is.close();
                }
            } catch (IOException e)
            {
                e.printStackTrace();
            }
        }
    }
}

你可能感兴趣的:(Java 你往我来的 IO 流,适合小白或傻傻分不清【输入、输出流】的伙计们 —— 第 03 篇 Java 的 字节输出缓冲流:BufferedOutputStream 字节输入缓冲流:Buff ..)