我的Java开发学习之旅------>Java 格式化类(java.util.Formatter)基本用法

时间:2020-12-23 04:34:01

我的Java开发学习之旅------>Java 格式化类(java.util.Formatter)基本用法

本文参考:

有时会想把数字,日期,字符串按照给定规则给格式化。SUN JDK为我们提供了这个API,它是java.util.Formatter。此类提供了对布局对齐和排列的支持,以及

对数值、字符串和日期/时间数据的常规格式和特定于语言环境的输出的支持。

如何格式化?

给定规则:

要想按照自己的想法格式化必须事先编写一个规则。那这个规则要怎么编写?

1.常规类型、字符类型和数值类型的格式说明符的语法如下:

%[argument_index$][flags][width][.precision]conversion

2.用来表示日期和时间类型的格式说明符的语法如下:

%[argument_index$][flags][width]conversion

3.与参数不对应的格式说明符的语法如下:

%[flags][width]conversion

API中有这样三种规则,很显然第一个规则的内容是最全面的。其它规则的内容和第一规则的内容有重复,那单说第一规则内容,其它规则依次类推。

注意:规则一中的precision前面要加英文句号“.”

API有以下解释:

可选的
argument_index
是一个十进制整数,用于表明参数在参数列表中的位置。第一个参数由 "1$"引用,第二个参数由"2$"引用,依此类推。

可选
flags
是修改输出格式的字符集。有效标志集取决于转换类型。

可选
width
是一个非负十进制整数,表明要向输出中写入的最少字符数。

可选
precision
是一个非负十进制整数,通常用来限制字符数。特定行为取决于转换类型。

所需
conversion
是一个表明应该如何格式化参数的字符。给定参数的有效转换集取决于参数的数据类型。

argument_index很好理解,就是一参数占位符,用来表示要被格式化的参数。

flags

标志

常规

字符

整数

浮点

日期/时间

说明

'-'

y

y

y

y

y

结果将是左对齐的。

'#'

y1

-

y3

y

-

结果应该使用依赖于转换类型的替换形式

'+'

-

-

y4

y

-

结果总是包括一个符号

' '

-

-

y4

y

-

对于正值,结果中将包括一个前导空格

'0'

-

-

y

y

-

结果将用零来填充

','

-

-

y2

y5

-

结果将包括特定于语言环境的组分隔符

'('

-

-

y4

y5

-

结果将是用圆括号括起来的负数

1 取决于Formattable
的定义。

2 只适用于 'd'转换。

3 只适用于 'o'、'x'和'X'转换。

4 对 BigInteger 应用 'd'、'o'、'x'和'X'转换时,或者对byte及Byte、short及Short、int及Integer、long及Long
分别应用 'd' 转换时适用。

5 只适用于 'e'、'E'、'f'、'g'和'G'转换。

任何未显式定义为标志的字符都是非法字符,并且都被保留,以供扩展使用。

width 就表示一最少字符数,被格式化参数用precision 截取器截取后与width 相比,被格式化参数字符数如果小于width ,则加字符则到等于width 。如果大于则width 不起作用。所以可以叫width 为少加多过器。

precision 在上面也提到了,precision 是一个截取器,用于截取被格式化参数。

conversion 转换可分为以下几类:

 常规-可应用于任何参数类型

 字符-可应用于表示Unicode字符的基本类型:char、Character、byte、Byte、short和Short。当Character.isValidCodePoint(int)
返回 true时,可将此转换应用于 int和Integer
类型

数值

1整数-可应用于Java的整数类型:byte、Byte、short、Short、int、Integer、long、Long
BigInteger

2浮点-可用于Java的浮点类型:float、Float、double、Double
BigDecimal

 日期/时间-可应用于Java的、能够对日期或时间进行编码的类型:long、LongCalendar
Date

 百分比-产生字面值'%' ('\u0025')

 行分隔符-产生特定于平台的行分隔符

常规

B b

如果参数arg 为 null,则结果为 "false"。如果arg
是一个 boolean 值或Boolean,则结果为String.valueOf()
返回的字符串。否则结果为 "true"。

H h

如果参数arg 为 null,则结果为 "null"。否则,结果为调用Integer.toHexString(arg.hashCode())
得到的结果。

S s

如果参数arg 为 null,则结果为 "null"。如果arg
实现 Formattable,则调用arg.formatTo。否则,结果为调用
arg.toString() 得到的结果。

字符

C c

结果是一个 Unicode字符

整数

d

结果被格式化为十进制整数

o

结果被格式化为八进制整数

X x

结果被格式化为十六进制整数

浮点

E e

结果被格式化为用计算机科学记数法表示的十进制数

f

结果被格式化为十进制数

G g

根据精度和舍入运算后的值,使用计算机科学记数形式或十进制格式对结果进行格式化。

A a

结果被格式化为带有效位数和指数的十六进制浮点数

日期,日间

T t

日期和时间转换字符的前缀

百分比

%

结果为字面值 '%'

行分隔符

n

结果为特定于平台的行分隔符

常出现的异常

java.util.IllegalFormatConversionException: d != java.lang.Double
: 被格式化参数类型与规则转换类型不对应。

java.util.FormatFlagsConversionMismatchException: Conversion = d, Flags = #  : flag不适用于规则转换类型。

Formatter类是用正则表达式验证给定规则的

正则表达式如下:

private static final
String formatSpecifier

=

"%(\\d+\\$)?([-#+0,(\\<]*)?(\\d+)?(\\.\\d+)?([tT])?([a-zA-Z%])";

总结:

  •    最重要的是它可以格式化日期/时间,数值和字符。
  •    可以把日期转换成年,月,日,星期等。可以为数值填充空格或0,添加分组字符,正负号,及小括号。我们可以在System.out.format   
    (),String.format()方法中直接应用Formatter类。

下面贴一段代码,将Android中的播放器的进度值之转换为相应的时间

/**
* 格式化的格式
*/
private StringBuilder sFormatBuilder = new StringBuilder();
/**
* 格式化类
*/
private Formatter sFormatter = new Formatter(sFormatBuilder, Locale.getDefault());
/**
* 格式化的相关属性
*/
private final Object[] sTimeArgs = new Object[3];
/**
* 转换进度值为时间
* @param secs
* @return
*/
private String makeTimeString(int secs) {
/**
* %[argument_index$][flags][width]conversion
* 可选的 argument_index 是一个十进制整数,用于表明参数在参数列表中的位置。第一个参数由 "1$" 引用,第二个参数由 "2$" 引用,依此类推。
可选 flags 是修改输出格式的字符集。有效标志集取决于转换类型。
可选 width 是一个非负十进制整数,表明要向输出中写入的最少字符数。
可选 precision 是一个非负十进制整数,通常用来限制字符数。特定行为取决于转换类型。
所需 conversion 是一个表明应该如何格式化参数的字符。给定参数的有效转换集取决于参数的数据类型。
*/
String durationformat = getString(R.string.durationformat);//<xliff:g id="format">%1$02d:%2$02d:%3$02d</xliff:g>
sFormatBuilder.setLength(0);
secs = secs / 1000;
Object[] timeArgs = sTimeArgs;
timeArgs[0] = secs / 3600; //时
timeArgs[1] = (secs % 3600) / 60; //分
timeArgs[2] = (secs % 3600 % 60) % 60; //秒
return sFormatter.format(durationformat, timeArgs).toString().trim();
}

其中R.string.durationformat定义如下:

<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
<string name="durationformat" translatable="false"><xliff:g id="format">%1$02d:%2$02d:%3$02d</xliff:g></string>
</resources>

关于格式化时间,还可以参考我的博客

《我的Java开发学习之旅------>工具类:将播放器的进度值转换成相应的时间格式》

地址:http://blog.csdn.net/ouyang_peng/article/details/17164255

什么是xliff?可以参考http://baike.baidu.com/view/4160854.htm

====================================================================================

  作者:欧阳鹏  欢迎转载,与人分享是进步的源泉!

  转载请保留原文地址:http://blog.csdn.net/ouyang_peng

====================================================================================