字节输出缓冲流: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();
}
}
}
}