当我检查以下代码的结果时,我感到难过:
public class StringBuilderTester {
public static void main(String[] args) {
StringBuilder matrix = new StringBuilder();
Random r = new Random();
Formatter formatter = new Formatter();
for (int j=0;j<10;j++) {
String[] strings = new String[10];
StringBuilder formatString = new StringBuilder();
for (int i=0;i<10;i++) {
formatString.append("%-10.5s");
strings[i] = new Integer(r.nextInt(10)).toString();
}
formatString.append("\n");
matrix.append(formatter.format(formatString.toString(), strings)); //gives me unexpected result: 1+2+3+..+10=55 lines of 10 cols of random ints
//matrix.append(String.format(formatString.toString(), strings)); //gives me expected results: 10 lines of 10 cols of random ints
}
System.out.println(matrix.toString());
}
}
我得到55(!)行,每列10个整齐格式的整数:第1行,第1行和第2行,然后是第1,2和3行,依此类推。似乎formatter.format
逐步发挥作用。
将formatter.format
更改为String.format
时,结果符合预期:10行整齐格式的整数。
Java参考说明了
可以使用静态方法
String.format
格式化字符串。
但没有迹象表明new Format().formatter(...)
可能不用于字符串。 (如果我使用构造函数Format(Appendable matrix)
,一切都按预期工作。)
这种不稳定行为的解释是什么?
答案 0 :(得分:2)
这两种情况实际上都是expected behaviour。 Formatter
类有一个内部缓冲区(默认为StringBuilder
),它将每次调用的结果追加到format
,toString()
方法返回整个内容缓冲区。所以你第一次打电话
matrix.append(formatter.format(formatString.toString(), strings));
它将row1添加到格式化程序缓冲区,然后将其附加到矩阵。再次调用它并将第2行添加到缓冲区,但现在缓冲区包含row1和row2,因此两行都会添加到矩阵中,现在它有第1行,第1行,第2行。这样就会继续并最终添加55行总数。
old matrix + buffer = new matrix
1 1
1 12 112
112 123 112123
112123 1234 1121231234
...and so on
另一方面,如果只调用String.format,它会产生与formatter.format
相同的输出,但不会将其附加到任何缓冲区。这样,在每次循环迭代中,只有最后创建的行被附加到矩阵,这是预期的结果。
TL; DR:formatter.format
将格式化的行添加到内部缓冲区,然后输出整个缓冲区,而String.format
只输出格式化的行。