Java 的格式化字符串

Java 的格式化字符串

一、格式化 Formatter

  • 参考了 C 语言的 printf() 函数,但 Java 的格式化会更严格一些
  • 同时定制化增加了 Java 语言的一些特性,使得格式化更适用于 Java 语言

二、格式化方法

2.1 使用 Formatter 实例

1
2
Formatter formatter = new Formatter();
formatter.format("%4$2s %3$2s %2$2s %1$2s", "a", "b", "c", "d")

2.2 使用 System 中的方法

1
2
3
4
5
// out
System.out.format("Local time: %tT", Calendar.getInstance());

// err
System.err.printf("Unable to open file '%1$s': %2$s", fileName, exception.getMessage());

2.3 使用 String 中的静态方法

1
String s = String.format("Duke's Birthday: %1$tb %1$te, %1$tY", c);

2.4 格式化方法要素

所有的格式化方法都有 2 个要素:

  • format string:格式化字符串
  • argument list:参数列表

比如:

1
String s = String.format("%4$2s %3$2s %2$2s %1$2s", "a", "b", "c", "d");

其中,

  • %4$2s %3$2s %2$2s %1$2s 是格式化字符串
  • "a", "b", "c", "d" 是参数列表

格式化字符串的写法是有固定语法的,不能随意写。

三、格式化语法

参数类型可以分为几类:

  • 通用:比如字符串
  • 字符:比如字母
  • 数值:比如整数和浮点数
  • 日期时间:比如年月日,时分秒
  • 无参类型:比如换行

对于不同的参数数据类型,格式化语法会稍微有点不一样。

3.1 通用、字符、数值

语法格式:

1
%[argument_index$][flags][width][.precision]conversion
  • argument_index:参数索引,指定使用第几个参数,1$ 表示第1个参数,2$ 表示第2个参数
  • flags:标志,用于调整参数输出的格式,比如左对齐,前缀补零等
  • width:输出宽度,用于指定参数输出的最小字符数量
  • precision:对于数值型参数,用于指定精度;对于某些类型,用于表示输出的最大字符数量
  • conversion:转换符,单个字母,用于表示参数如何被格式化

举个例子:

1
2
3
4
5
("%4$2s", "a", "b", "c", "d") --> " d"

- 4$: argument_index,这里表示使用第 4 个参数
- 2: width,这里表示输出字符数量最少为 2
- s: conversion,这里表示以字符串形式输出参数,即 toString()

3.2 日期、时间

语法格式:

1
%[argument_index$][flags][width]conversion
  • argument_indexflagswidth 和上面的通用型含义一样
  • conversion:转换符,2 个字母,第一个字母是 tT,用于表示参数如何被格式化

举个例子:

1
2
3
4
5
("%1$tY", new Date()) --> 2023

- 1$: argument_index,这里表示使用第 1 个参数
- t: conversion,日期和时间转换符的固定前缀
- Y: conversion,年份,最少 4 位整数表示,不足 4 位时前面补零

3.3 无参类型

语法格式:

1
%[flags][width]conversion
  • flagswidth 和上面的通用型含义一样
  • conversion:表示输出的内容,没有参数,不是占位符了

举个例子:

1
2
3
("%d%%", 50) --> "50%"

- %: 表示输出 % 这个字符

四、转换符(conversion)

4.1 转换符分类

类型 应用范围 适用参数类型
通用型 所有参数类型
字符型 可应用于 Unicode 字符展示的基本类型 包括 charCharacterbyteByteshortShort
还有可能应用于 intInteger 类型(满足 Character.isValidCodePoint
数值型 可分为整型和浮点型 整型包括 byteByteshortShortintIntegerlongLongBigInteger
浮点型包括 floatFloatdoubleDoubleBigDecimal
日期时间 可应用于日期和时间 包括 longLongCalendarDateTemporalAccessor
百分比 特定字面量 % 只有 %
换行符 特定平台的换行符 比如 \n\r\n

4.2 转换符汇总

转换符 分类类型 描述
b/B 通用型 布尔值。
null 的结果是 false
布尔参数时等于 String.valueOf(arg)
非布尔参数时等于 true
h/H 通用型 哈希值。
结果等于 Integer.toHexString(arg.hashCode())
s/S 通用型 字符串。
如果参数实现了 Formattable 接口,那么输出就是 arg.formatTo() 的结果
否则输出等于 arg.toString()
c/C 字符型 Unicode 字符
d 整型 整数
o 整型 八进制整数
x/X 整型 十六进制整数
e/E 浮点型 科学计数法
f 浮点型 小数
g/G 浮点型 小数或者科学计数法,根据数据的实际情况动态决定使用哪种
a/A 浮点型 十六进制浮点数,带有效数和指数
t/T 日期时间 日期时间转换符的固定前缀
% 百分比 字面量 %
n 换行符 特定平台的换行符

转换符大小写的格式化作用相同,仅仅是输出结果的大小写不一样。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
/* 布尔值 b/B */
System.out.printf("null: %b, %B", null, null);
// null: false, FALSE
System.out.printf("bool: %b, %B", true, false);
// bool: true, FALSE
System.out.printf("other: %b, %B", 1, "test");
// other: true, TRUE

/* 哈希值 h/H */
System.out.printf("null: %h, %H", null, null);
// null: null, NULL
System.out.printf("hash: %h, %H", 12.0, "false");
// hash: 40280000, 5CB1923

/* 字符串 s/S */
System.out.printf("null: %s, %S", null, null);
// null: null, NULL
System.out.printf("string: %s, %S", 123, "string");
// string: 123, STRING

/* 字符 c/C */
System.out.printf("null: %c, %C", null, null);
// null: null, NULL
System.out.printf("string: %c, %C", 36, 'a');
// char: $, A

/* 整数值 d/o/x/X */
System.out.printf("null: %d", null);
// null: null
System.out.printf("10-int: %d", 36);
// 10-int: 36
System.out.printf("8-int: %o", 40);
// 8-int: 50
System.out.printf("16-int: %x, %X", 46, 47);
// 16-int: 2e, 2F

/* 浮点值 e/E/f/g/G/a/A */
System.out.printf("null: %f", null);
// null: null
System.out.printf("scientific: %e, %E", 360000000000000.0, 0.00002);
// scientific: 3.600000e+14, 2.000000E-05
System.out.printf("float: %f, %f", 40.0f, 100.2);
// float: 40.000000, 100.200000
System.out.printf("rounding: %g, %G", 0.0001, 0.00000000000001);
// rounding: 0.000100000, 1.00000E-14
System.out.printf("16-float: %a, %A %n", 0.0001, 0.00000000000001);
// 16-float: 0x1.a36e2eb1c432dp-14, 0X1.6849B86A12B9BP-47

/* 日期时间 t/T */
System.out.printf("null: %th", null);
// null: null
System.out.printf("year: %ty, %tY", new Date(), new Date());
// year: 23, 2023
System.out.printf("month: %tM", new Date());
// month: 09
System.out.printf("hour: %th, %tH", new Date(), new Date());
// hour: 01
System.out.printf("16-float: %s, %S", new Date(), new Date());
// second: 1673025096, 36

/* 百分比 % */
System.out.printf("percent: %%");
// percent: %

/* 换行符 n */
System.out.printf("line separator: %n line separator");
//line separator:
// line separator

4.3 日期转换符

转换符 描述 示例
Y 4位数的年份 0092
y 2位数的年份 范围是 00 - 99
B 本地化的月份全称 “January”、”February”
b 本地化的月份简称 “Jan”、”Feb”
h 和 b 一样
m 2位数的月份,从01开始 范围是 01 - 13
A 本地化的周全称 “Sunday”、”Monday”
a 本地化的周简称 “Sun”、”Mon”
d 2位数的每月天数 范围是 01 - 31
e 每月天数 范围是 1 - 31
1
2
3
4
5
6
7
8
9
10
System.out.printf("year: %ty, %tY", new Date(), new Date());
// year: 23, 2023
System.out.printf("month name: %tb, %tB, %th", new Date(), new Date(), new Date());
// month name: 1月, 一月, 1月
System.out.printf("month number: %tm", new Date());
// month number: 01
System.out.printf("week: %ta, %tA", new Date(), new Date());
// week: 周六, 星期六
System.out.printf("day: %td, %te", new Date(), new Date());
// day: 07, 7

4.4 时间转换符

转换符 描述
H 小时,24小时制,2位数,范围是 00 - 23
I(i的大写) 小时,12小时制,2位数,范围是 01 - 12
k 小时,24小时制,范围是 0 - 23
l(L的小写) 小时,12小时制,范围是 1 - 12
M 分钟,2位数,范围是 00 - 59
S 秒,2位数,范围是 00 - 60,60专门用于支持闰秒
L 毫秒,3位数,范围是 000 -999
N 纳秒,9位数,范围是 000000000 - 999999999
s 从 1970/01/01 00:00:00 开始的秒数,范围是 Long.MIN_VALUE/1000 - Long.MAX_VALUE/1000
Q 从 1970/01/01 00:00:00 开始的毫秒数,范围是 Long.MIN_VALUE - Long.MAX_VALUE
p 特定语言的上午或者下午,比如 am,pm
1
2
3
4
5
6
7
8
9
10
11
12
System.out.printf("hour: %tH, %tI, %tk, %tl", date, date, date, date);
// hour: 21, 09, 21, 9
System.out.printf("minute: %tM", date);
// minute: 33
System.out.printf("second: %ts, %tS", date, date);
// second: 1673012005, 25
System.out.printf("millisecond: %tL, %tQ", date, date);
// millisecond: 706, 1673012005706
System.out.printf("nanosecond: %tN", date);
// nanosecond: 706000000
System.out.printf("noon: %tp", date);
// noon: 下午

上面是比较细的转换符,直接用的话写起来比较麻烦。

而且日期时间的输出还是很常用的,所以还提供了一些时间转换符的快捷方式:

转换符 描述 示例
R 表示24小制时分:%tH:%tM 范围是 00:00 - 23:59
T 表示24小制时分秒:%tH:%tM:%tS 范围是 00:00:00 - 23:59:59
D 表示日期的月日年:%tm/%td/%ty 比如 01/06/23
F 表示日期的年月日:%tY-%tm-%td 比如 2023-01-06
c 完整的时间日期:%ta %tb %td %tT %tZ %tY 比如 Sun Jul 20 16:17:00 EDT 1969
1
2
3
4
5
6
7
8
9
10
System.out.printf("HH:MM: %tR", new Date());
// HH:MM: 01:38
System.out.printf("HH:MM:SS: %tT", new Date());
// HH:MM:SS: 01:38:51
System.out.printf("mm/dd/yy: %tD", new Date());
// mm/dd/yy: 01/07/23
System.out.printf("YYYY-mm-dd: %tF", new Date());
// YYYY-mm-dd: 2023-01-07
System.out.printf("date time: %tc", new Date());
// date time: 周六 1月 07 01:38:51 CST 2023

时间转换符的快捷方式就是用简单的1个字符替代多个转换符,简化代码,相当于缩写。

五、标志位(flags)

标志位用于对输出格式进行一些调整。

标志 描述 示例 结果
- 左对齐 (“^%-5d^”, 15) ^15   ^
# 如果是浮点数则包含小数点,如果是16进制或8进制则添加0x或0 (“%#x”, 99) 0x63
+ 给数值加上符号位,正数前加+,负数前加- (“%+d”, 99) +99
空格 在整数之前添加指定数量的空格 (“^% 4d^”, 99) ^  99^
0 在数字前面补0 (“%04d”, 99) 0099
, 以“,”对数字分组 (“%,d”, 999999999) 999,999,999
( 使用括号包含负数 (“%(f”, -99.99) (99.99)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
System.out.printf("left-justified: ^%5d^, ^%-5d^", 99, 99);
// left-justified: ^ 99^, ^99 ^
System.out.printf("conversion-dependent: %o, %#o", 40, 40);
// conversion-dependent: 50, 050
System.out.printf("conversion-dependent: %X, %#X", 47, 47);
// conversion-dependent: 2F, 0X2F
System.out.printf("sign: %d, %+d", 99, 99);
// sign: 99, +99
System.out.printf("space: ^%d^, ^% 5d^", 99, 99);
// space: ^99^, ^ 99^
System.out.printf("zero: ^%4d^, ^%04d^", 99, 99);
// zero: ^ 99^, ^0099^
System.out.printf("grouping separators: %d, %,d", 123456789, 123456789);
// grouping separators: 123456789, 123,456,789
System.out.printf("negative parentheses: %d, %(d", -99, -99);
// negative parentheses: -99, (99)

六、宽度(width)

宽度的含义:

  • 表示输出字符数量的最小值
  • 类型是整数,范围是 1 - Integer.MAX_VALUE
1
2
3
4
System.out.printf("width: ^%4d^", 99);
// width: ^ 99^
System.out.printf("width: %4s", "123456789");
// width: 123456789

因为输出字符数量最小是 4,因此在 99 之前补了2个空格,所以 ("^%4d^", 99) 的结果是 ^ 99^

七、精度(precision)

精度对于不同类型的含义:

  • 对于通用型,精度表示输出字符数量的最大值
  • 对于浮点型,精度表示小数点后面的数字个数
  • 对于字符型、整型、日期时间、百分比、换行符等,精度不支持
1
2
3
4
System.out.printf("general: %.6s", "123456789");
// general: 123456
System.out.printf("float: %.3f", 123.342235625345);
// float: 123.342

参考

java.util.Formatter

https://blog.csdn.net/lonely_fireworks/article/details/7962171/

作者

jiaduo

发布于

2023-01-08

更新于

2023-04-02

许可协议