Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >面试官:Java中缓冲流真的性能很好吗?我看未必

面试官:Java中缓冲流真的性能很好吗?我看未必

原创
作者头像
JavaBuild
发布于 2024-06-16 12:08:54
发布于 2024-06-16 12:08:54
1400
举报
文章被收录于专栏:Java成长计划Java成长计划

一、写在开头

上一篇文章中,我们介绍了Java IO流中的4个基类:InputStream、OutputStream、Reader、Writer,那么这一篇中,我们将以四个基类所衍生出来,应对不同场景的数据流进行学习。

二、衍生数据流分类

我们上面说了java.io包中有40多个类,都从InputStream、OutputStream、Reader、Writer这4个类中衍生而来,我们以操作对象的维度进行如下的区分:

2.1 文件流

文件流也就是直接操作文件的流,可以细分为字节流(FileInputStream 和 FileOuputStream)和字符流(FileReader 和 FileWriter),我们在上面的已经说了很多了,这里就再赘述啦。

2.2 数组流

所谓数组流就是将内存中有限的数据进行读写操作的流,适应于数据量小,无需利用文件存储,提升程序效率。

我们以ByteArrayInputStream(字节数组输入流)为例:

代码语言:java
AI代码解释
复制
public class TestService{
    public static void main(String[] args)  {
        try {
            ByteArrayInputStream bi = new ByteArrayInputStream("JavaBuild".getBytes());
            int content;
            while ((content = bi.read()) != -1) {
                System.out.print((char) content);
            }
            // 关闭输入流,释放资源
            bi.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

字节数组输出流(ByteArrayOutputStream)亦是如此,它们不需要创建临时文件,直接在内存中就可以完成对字节数组的压缩,加密,读写以及序列化。

2.3 管道流

管道(Pipe)作为一种在计算机内通讯的媒介,无论是在操作系统(Unix/Linux)层面还是JVM层面都至关重要,我们今天提到的通道流就是在JVM层面,同一个进程中不同线程之间数据交互的载体。

我们以PipedOutputStream和PipedInputStream为例,通过PipedOutputStream将一串字符写入到内存中,再通过PipedInputStream读取输出到控制台,整个过程并没有临时文件的事情,数据仅在两个线程之间流转。

代码语言:java
AI代码解释
复制
public class TestService{
    public static void main(String[] args) throws IOException {
        // 创建一个 PipedOutputStream 对象和一个 PipedInputStream 对象
        final PipedOutputStream pipedOutputStream = new PipedOutputStream();
        final PipedInputStream pipedInputStream = new PipedInputStream(pipedOutputStream);

        // 创建一个线程,向 PipedOutputStream 中写入数据
        Thread thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    // 将字符串 "沉默王二" 转换为字节数组,并写入到 PipedOutputStream 中
                    pipedOutputStream.write("My name is JavaBuild".getBytes());
                    // 关闭 PipedOutputStream,释放资源
                    pipedOutputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        });
        // 创建一个线程,从 PipedInputStream 中读取数据并输出到控制台
        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    // 定义一个字节数组用于存储读取到的数据
                    byte[] flush = new byte[1024];
                    // 定义一个变量用于存储每次读取到的字节数
                    int len = 0;
                    // 循环读取字节数组中的数据,并输出到控制台
                    while (-1 != (len = pipedInputStream.read(flush))) {
                        // 将读取到的字节转换为对应的字符串,并输出到控制台
                        System.out.println(new String(flush, 0, len));
                    }
                    // 关闭 PipedInputStream,释放资源
                    pipedInputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        });
        // 启动线程1和线程2
        thread1.start();
        thread2.start();
    }
}

2.4 数据流

我们知道在Java中分为基本数据类型和引用类型,我们在做数据的读取与写入时,自然也会涉及到这种情况,比如我们将txt文件中的数字型数据以int类型读取到程序中,这时Java为我们提供了DataInputStream/DataOutputStream类。它们的常用方法为:

具体使用也相对比较简单:

代码语言:java
AI代码解释
复制
DataInputStream dis = new DataInputStream(new FileInputStream("input.txt"));
// 创建一个 DataOutputStream 对象,用于将数据写入到文件中
DataOutputStream das = new DataOutputStream(new FileOutputStream("output.txt"));
// 读取四个字节,将其转换为 int 类型
int i = dis.readInt();
// 将一个 int 类型的数据写入到文件中
das.writeInt(1000);

2.5 缓冲流

对于数据的处理,CPU速度快于内存,内存又远快于硬盘,在大数据量情况下,频繁的通过IO向磁盘读写数据会带来严重的性能问题,为此Java中提供了一个缓冲流的概念,简单来说就是在内存中设置一个缓冲区,只有缓冲区中存储的数据到达一定量后才会触发一次IO,这样大大提升了程序的读写性能,常用的缓冲流有:BufferedInputStream、BufferedOutputStream、BufferedReader、BufferedWriter。

通过BufferedInputStream的底层源码我们可以看到,其内部维护了一个buf[]数据,默认大小为8192字节,我么也可以通过构造函数进行缓存大小设置。

代码语言:java
AI代码解释
复制
public
class BufferedInputStream extends FilterInputStream {
    // 内部缓冲区数组
    protected volatile byte buf[];
    // 缓冲区的默认大小
    private static int DEFAULT_BUFFER_SIZE = 8192;
    // 使用默认的缓冲区大小
    public BufferedInputStream(InputStream in) {
        this(in, DEFAULT_BUFFER_SIZE);
    }
    // 自定义缓冲区大小
    public BufferedInputStream(InputStream in, int size) {
        super(in);
        if (size <= 0) {
            throw new IllegalArgumentException("Buffer size <= 0");
        }
        buf = new byte[size];
    }
}

至于说缓冲流到底能不能实现性能的提升,我们实践出真知,对于程序员来说所有的理论都不及上手写一写来得有效!这其实也涉及到一个经常被问的面试问题:java中的缓冲流真的性能很好吗?

刚好,我们手头有一本《Java性能权威指南》的PDF版,大小为66MB,我们通过普通的文件流和缓冲流进行文件的读取和复制,看一下耗时对比。

代码语言:java
AI代码解释
复制
public class TestService{
    public static void main(String[] args) throws IOException {
        TestService testService = new TestService();
        testService.copyPdfWithPublic();
        testService.copyPdfWithBuffer();
    }
    /*通过普通文件流进行pdf文件的读取和拷贝*/
    public void copyPdfWithPublic(){
        // 记录开始时间
        long start = System.currentTimeMillis();
        try (FileInputStream fis = new FileInputStream("E:\\Java性能权威指南.pdf");
             FileOutputStream fos = new FileOutputStream("E:\\Java性能权威指南Public.pdf")) {
            int content;
            while ((content = fis.read()) != -1) {
                fos.write(content);
            }
            //使用数组充当缓存时,两者性能差距不大
            /*int len;
            byte[] bytes = new byte[4 * 1024];
            while ((len = fis.read(bytes)) != -1) {
                fos.write(bytes, 0, len);
            }*/
            fis.close();
            fos.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        // 记录结束时间
        long end = System.currentTimeMillis();
        System.out.println("使用普通文件流复制PDF文件总耗时:" + (end - start) + " 毫秒");
    }
    /*通过缓冲字节流进行pdf文件的读取和拷贝*/
    public void copyPdfWithBuffer(){
        // 记录开始时间
        long start = System.currentTimeMillis();
        try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream("E:\\Java性能权威指南.pdf"));
             BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("E:\\Java性能权威指南Buffer.pdf"))) {
            int content;
            while ((content = bis.read()) != -1) {
                bos.write(content);
            }
            /*int len;
            byte[] bytes = new byte[4 * 1024];
            while ((len = bis.read(bytes)) != -1) {
                bos.write(bytes, 0, len);
            }*/
            bis.close();
            bos.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        // 记录结束时间
        long end = System.currentTimeMillis();
        System.out.println("使用缓冲字节流复制PDF文件总耗时:" + (end - start) + " 毫秒");
    }
}

输出:

代码语言:java
AI代码解释
复制
使用普通文件流复制PDF文件总耗时:221611 毫秒
使用缓冲字节流复制PDF文件总耗时:228 毫秒

然后,我们将注释掉的代码放开,也就是我们采用一个缓存数组,先将数组存储起来后,两者之间的性能差距就没那么明显了。

代码语言:java
AI代码解释
复制
使用普通文件流复制PDF文件总耗时:106 毫秒
使用缓冲字节流复制PDF文件总耗时:80 毫秒

在这种情况下,我们可以看到,甚至于普通的文件流的耗时是小于缓冲流的,所以对于这种情况来说,缓冲流未必一定性能最好。

2.6 打印流

对于System.out.println("Hello World");这句代码我想大家并不陌生吧,我们刚学习Java的第一堂课,老师们都会让我们输出一个Hello World,System.out 实际是用于获取一个 PrintStream 对象,print方法实际调用的是 PrintStream 对象的 write 方法。

代码语言:java
AI代码解释
复制
public class PrintStream extends FilterOutputStream
    implements Appendable, Closeable {
}
public class PrintWriter extends Writer {
}

我正在参与2024腾讯技术创作特训营最新征文,快来和我瓜分大奖!

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
Java IO 知识整理
由于java I/O库需要很多性能的各种组合,如果这些性能都是用继承来实现,那么每一种组合都需要一个类,这样就会造成大量行重复的类出现。如果采用装饰模式,那么类的数目就会大大减少,性能的重复也可以减至最少。因此装饰模式是java I/O库基本模式。装饰模式的引进,造成灵活性和复杂性的提高。因此在使用java IO库时,必须理解java IO库是由一些基本的原始流处理器和围绕它们的装饰流处理器所组成的。
Abalone
2022/07/14
5920
Java IO 知识整理
【Java 基础篇】Java字节缓冲流详解
在Java中,字节缓冲流是一种用于提高字节流读写效率的流。它们通过在内存中创建缓冲区,减少了与底层设备的直接交互次数,从而提高了读写的速度。本文将详细介绍Java字节缓冲流的原理、使用场景和常用类,并提供一些示例代码。
繁依Fanyi
2023/10/12
4500
Java成神路 —— 字节缓冲流 && 字符流
0. 字符流与字节流 1.字节缓冲流 1.1 字节缓冲流构造方法 1. 字节缓冲流介绍 lBufferOutputStream:该类实现缓冲输出流。 通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用 lBufferedInputStream:创建BufferedInputStream将创建一个内部缓冲区数组。 当从流中读取或跳过字节时,内部缓冲区将根据需要从所包含的输入流中重新填充,一次很多字节 2. 构造方法 方法名 说明 BufferedOutputS
繁依Fanyi
2023/05/07
3710
Java成神路 —— 字节缓冲流 && 字符流
夯实Java基础系列16:一文读懂Java IO流和常见面试题
本系列文章将整理到我在GitHub上的《Java面试指南》仓库,更多精彩内容请到我的仓库里查看
Java技术江湖
2019/10/08
1.5K0
卧槽,这也太上头了吧!
“老王,Java IO 也太上头了吧?”新兵蛋子小二向头顶很凉快的老王抱怨道,“你瞧,我就按照传输方式对 IO 进行了一个简单的分类,就能搞出来这么多的玩意!”
沉默王二
2021/11/16
3490
卧槽,这也太上头了吧!
梳理Java IO流,攻破NIO的堡垒!【技术创作101训练营】
在上一篇的文章获取不错的浏览量后,继续加更的念头一直徘徊在心中,本来是想花段时间深入学习tomcat的,可是tomcat的源码中就有至关重要的NIO,于是得先整一下NIO,但是NIO的基础是BIO,于是这篇文章就先写IO流吧。
猿小布
2020/09/23
6340
梳理Java IO流,攻破NIO的堡垒!【技术创作101训练营】
【愚公系列】2022年01月 Java教学课程 54-数据缓冲
lBufferOutputStream:该类实现缓冲输出流.通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用
愚公搬代码
2022/01/15
2120
【愚公系列】2022年01月 Java教学课程 54-数据缓冲
JavaSE(十二)之IO流的字节流(一)
前面我们学习的了多线程,今天开始要学习IO流了,java中IO流的知识非常重要。但是其实并不难,因为他们都有固定的套路。 一、流的概念       流是个抽象的概念,是对输入输出设备的抽象,Java程序中,对于数据的输入/输出操作都是以"流"的方式进行。设备可以是文件,网络,内存等       流具有方向性,至于是输入流还是输出流则是一个相对的概念,一般以程序为参考,如果数据的流向是程序至设备,我们成为输出流,如果数据的流向是设备至程序称为输入流。       数据以二进制的形式在程序与设备之间流动传输,就
用户1195962
2018/01/18
9490
JavaSE(十二)之IO流的字节流(一)
Java I/O 常用的实现类
FileInputStream类可以将一个文件的内容作为字节流读取,我们看一下源码:
搬砖俱乐部
2019/06/16
9750
java常用的io流_io流java
IO流大家肯定不陌生,简单整理了一下常用IO流基本用法,其他的IO流还有很多以后有时间在整理。
全栈程序员站长
2022/11/01
1.7K0
java常用的io流_io流java
java中IO流详细解释?
BufferedReader的readLine()&BufferedWriter的newLine()方法
技术从心
2019/08/07
1.8K0
Java基础17:Java IO流总结
本文介绍了Java IO流的基本概念,使用方法,以及使用的注意事项等。帮助你更好地理解和使用Java的IO流。
程序员黄小斜
2019/04/07
1K0
Java基础入门笔记08——文件流,IO流,FileIputStream类,FileOutputStream类,缓冲流,转换流,标准输入输出流,打印流,数据流,对象流,随机存储流
**绝对路径:**直接指明文件在硬盘上真正存在具体位置或者是以Web站点根目录为参考的完整路径。绝对路径是规定死的目录,直观清晰,但被网页引用的文件不能随意挪动。当多个网页引用同一个文件时,所使用的路径都是相同的。
全栈程序员站长
2022/09/21
1.6K0
Java基础入门笔记08——文件流,IO流,FileIputStream类,FileOutputStream类,缓冲流,转换流,标准输入输出流,打印流,数据流,对象流,随机存储流
java缓冲流
使用缓冲流的好处是,能够更高效的读写信息,原理是将数据先缓冲起来,然后一起写入或者读取出来。经常使用的是readLine()方法,表示一次读取一行数据。
IT工作者
2022/05/09
8720
Java IO 的分类、使用场景、缓冲流、使用例子、文件拷贝
Java程序在执行的时候,是在内存进行的,外部的数据需要读写到内存才能处理;而在内存中的数据是随着程序结束就消失的,有时候我们也需要把数据输出到外部文件。
HaC
2021/02/04
1.6K0
Java IO 的分类、使用场景、缓冲流、使用例子、文件拷贝
面试被虐,临走时面试官甩给我这篇介绍IO的文章
现在很多项目的开发都会用到SpringBoot,而SpringBoot的自动配置的底层原理实现就在于,Java提供的IO流将META-INF/spring-autoconfigure-metadata.properties文件中的数据读取出来,所以适当了解Java的IO流是十分有必要的。本文的介绍内容如下:
Java_老男孩
2020/06/28
4860
Java IO流学习总结三:缓冲流-BufferedInputStream、BufferedOutputStream
Java IO流学习总结一:输入输出流 Java IO流学习总结二:File Java IO流学习总结三:缓冲流-BufferedInputStream、BufferedOutputStream Java IO流学习总结四:缓冲流-BufferedReader、BufferedWriter Java IO流学习总结五:转换流-InputStreamReader、OutputStreamWriter Java IO流学习总结六:ByteArrayInputStream、ByteArrayOutputStream Java IO流学习总结七:Commons IO 2.5-FileUtils
全栈程序员站长
2022/11/01
9310
Java IO流学习总结三:缓冲流-BufferedInputStream、BufferedOutputStream
14 Java IO流
创建一个 FileWriter 对象,该文件会在指定目录下创建. 如果同名则覆盖, 除非构造方法第二个参数append 为 true; 由此可得默认为false.
acc8226
2022/05/17
4460
14 Java IO流
Java学习笔记之字节缓冲流&字符流&IO流练习
    BufferOutputStream:该类实现缓冲输出流,通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节频繁调用系统底层
Jetpropelledsnake21
2021/12/31
5610
Java学习笔记之字节缓冲流&字符流&IO流练习
IO流
不同设备之间的数据传输就是流,可以抽象理解供水厂到家庭之间的水管,水(数据)在管道传输,就成了流
晚上没宵夜
2020/03/11
4020
相关推荐
Java IO 知识整理
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档