【Java】System.out.flush()不起作用/没有效果

在学习Java编程时,发现`System.out.flush()`并未按预期立即刷新输出,导致输出内容仍以缓冲区的方式一组一组显示。文章讨论了缓冲区的概念及其在不同语言中的刷写策略,并分享了一段期望每30毫秒输出一个字符但实际并未即时显示的代码示例。尽管尝试使用`flush()`,问题依旧存在,暗示Java的刷写机制可能与预想不同。文中还附带了JVM的相关信息,揭示可能的刷新间隔为220ms。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

学Java的时候边学边写代码,遇到一个非常奇怪的问题。

众所周知,一般高级语言的一般输出语句实际上都并不能保证将内容实时显示到标准输出(如C/++的printf(),Python的print(),等等等等),而是写入一个叫做缓冲区(一般称为buffer)的一段内存空间。输入有输入buffer,输出有输出buffer,大多数的文件操作符都会定义一个自己的buffer。1
buffer的好处不用多说,由于buffer在内存上,可以随机存取,速度比硬盘之类的存储介质快几个数量级;但是buffer也有弱点,那就是刷写策略的问题。
一般在调用诸如printf()时,由于C/++的刷写策略有一条是遇到换行符\n就进行一次刷写。Python也有类似的策略,你可以通过指定flush=True来让print()函数立即刷写buffer到文件。

重头戏来了:学Java的输出语句System.out.printf()的时候,我从代码补全看到System.out.flush(),本以为这就是Java的刷写函数,结果
完 全 不 是 这 么 回 事 ! ! !


看下这段代码

import java.io.*;
import static java.lang.Math.round;

public class aClass {
    public static void main(String[] args) {

        int mod=26;
        
        for (int i = 0; i < 100; i++) {
            System.out.printf("%c", (char) (i % mod + 97));
            System.out.flush();
            try {
                Thread.sleep(30);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

显然这段代码“希望得到”一长串的abcdefgh...的内容,且每隔30毫秒输出一个字符。
然而…
当我运行这段代码时,问题发生了:字符还是一组一组地蹦出来!好像System.out.flush()完全什么都没有做一样
显然System.outBufferedOutputStream,但它仍然不可以flush(),截止现在我仍然不知道这是为什么。


贴出我的JVM信息:

VM=Java HotSpot(TM) 64-Bit Server VM,
vendor=Oracle Corporation,
version=25.121-b13

这段代码进行了一些修改,可以看出大约每隔220ms会自动flush一次System.out

public class aClass {
    public static void main(String[] args) {

        long now = System.currentTimeMillis(), d = 0, avg = 0;

        int mod = 7;  //换行,方便看出来每周期打多少个
        for (int i = 0; i < 100; i++) {
            if (mod - 1 == (i % mod)) {
                d = System.currentTimeMillis() - now;  //用来计时
                avg = round(avg * (int) (i / mod) + d) / (int) (i / mod + 1);  //求平均flush时间间隔
                System.out.printf("       d(now)=%d, avg=%d\n", d, avg);
                now = System.currentTimeMillis();
            }

            System.out.printf("%c", (char) (i % mod + 97));
            System.out.flush();
            try {
                Thread.sleep(30);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        //打印信息
        System.out.printf("\n===============\nVM=%s,\nvendor=%s,\nversion=%s\n", System.getProperty("java.vm.name"), System.getProperty("java.vm.vendor"),System.getProperty("java.vm.version"));

    }
}

  1. 有关于缓冲区(buffer)和缓存(cache)的区别,可以参考这里 ↩︎

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值