学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.out
是BufferedOutputStream
,但它仍然不可以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"));
}
}