【原文參考鏈接】
String.format()方法使用說明
JAVA字符串格式化-String.format()的使用
java的String.format的一些用法和Android中double顯示成科學(xué)計(jì)數(shù)的問題
JDK1.5開始String類中提供了一個(gè)非常有用的方法String.format(String format, Object ... args)
查看源碼得知其實(shí)是調(diào)用了Java.util.Formatter.format(String, Object...)方法
public static String format(String format, Object ... args) {
return new Formatter().format(format, args).toString();
}
首先來看一個(gè)例子:
String s2 = String.format("%1$tY-%1$tm-%1$te", new Date());
System.out.println(s2);
這里會(huì)打印出什么內(nèi)容?
查看JDK文檔得知,String.format方法的第一個(gè)參數(shù)是有個(gè)公式可以套的
%[argument_index$][flags][width][.precision]conversion
這里我們只要牢記這個(gè)公式就可以,下面說下每個(gè)參數(shù)所代表的含義:
-
argument_index:可選,是一個(gè)十進(jìn)制整數(shù),用于表明參數(shù)在參數(shù)列表中的位置。第一個(gè)參數(shù)由1$引用,第二個(gè)參數(shù)由2$引用,依此類推。 -
flags:可選,用來控制輸出格式 -
width:可選,是一個(gè)正整數(shù),表示輸出的最小長度 -
precision:可選,用來限定輸出字符數(shù) -
conversion:必須,用來表示如何格式化參數(shù)的字符
先看一個(gè)簡單的列子:
System.out.println(String.format("我的名字叫%s", "小明"));
// 打印:我的名字叫小明
這里我們只用了%s這個(gè)簡單的表達(dá)式,對(duì)比上面的公式,我們發(fā)現(xiàn)[argument_index$][flags][width][.precision]這些部分全部都省略掉了,只留下一個(gè)必須的conversion,在這里conversion就是s,百分號(hào)%是固定不變的
[argument_index$]省略之后它會(huì)自動(dòng)把"小明"這個(gè)值填入到%s中去
我再稍微改下例子:
String.format("我叫%s,她叫%s", "小明","小方"); // 我叫小明,她叫小方 `
這里會(huì)按順序分別把小明,小方填入到對(duì)應(yīng)的%s中. 如果我們要把小方填在前面,小明填在后面,那該怎么做呢,[argument_index$]就派上用場了
String.format("我叫%2$s,她叫%1$s", "小明","小方"); // 我叫小方,她叫小明 `
依然是百分號(hào)%開頭,中間多了個(gè)2$,1$
conversion可以填s,那還有什么其它字母可以填呢,當(dāng)然有的:
- o:結(jié)果被格式化為八進(jìn)制整數(shù)
- x:結(jié)果被格式化為十六進(jìn)制
- d:結(jié)果被格式化為十進(jìn)制整數(shù)
System.out.println(String.format("%o", 8)); // 10
System.out.println(String.format("%x", 16)); // 10
更多的conversion類別可以參考JDK文檔java.util.Formatter類
至此,我們已經(jīng)了解了argument_index$和conversion的用處,接下來我們了解flag和width的用法
-
flag是用來控制輸出格式的,比如左對(duì)齊,金額用逗號(hào)隔開等 -
width表示最小寬度
先看個(gè)列子:
String.format("%1$,d", 12302562); // 12,302,562
這里多出一個(gè)逗號(hào)",",它就是flag,用于金額千分位隔開,當(dāng)然寫成"%,d"也是可以的
再一個(gè)列子:
String.format("%1$08d", 123456);// 00123456
這里0就是flag,表示結(jié)果將用零來填充,8就是width,表示最少要8位,d是conversion
至于其它的flag可以查閱JDK文檔。
接下來說下[.precision]
這個(gè)單詞翻譯下是精度的意思,我們發(fā)現(xiàn)了前面有個(gè)小數(shù)點(diǎn).,因此不難聯(lián)想到這個(gè)是關(guān)于浮點(diǎn)數(shù)類型的。
只有當(dāng)傳入的數(shù)據(jù)是浮點(diǎn)數(shù)時(shí)這個(gè)才有用,整數(shù)或者日期類型的數(shù)據(jù)都不能用。
比如我想要四舍五入保留兩位小數(shù),那么我可以這么寫:
String.format("%1$.2f", 12.12555); //12.13
這里f表示傳入的數(shù)字是浮點(diǎn)型,如果傳入的是整數(shù),或者把f改成d都會(huì)拋出異常,JDK文檔中有明確說明
對(duì)于浮點(diǎn)轉(zhuǎn)換'e'、'E'和'f',精度是小數(shù)點(diǎn)分隔符后的位數(shù)。如果轉(zhuǎn)換是'g'或'G',那么精度是舍入計(jì)算后所得數(shù)值的所有位數(shù)。如果轉(zhuǎn)換是'a'或'A',則不必指定精度。
對(duì)于字符、整數(shù)和日期/時(shí)間參數(shù)類型轉(zhuǎn)換,以及百分比和行分隔符轉(zhuǎn)換,精度是不適用的;如果提供精度,則會(huì)拋出異常。
到現(xiàn)在為止這套表達(dá)式公式已經(jīng)基本講完了,這套公式是針對(duì)于基本數(shù)據(jù)類型,和字符串的,如果是正對(duì)于時(shí)間類型的數(shù)據(jù)該怎么做呢,比如格式化日期
其實(shí)文檔中已經(jīng)給出說明了:
用來表示日期和時(shí)間類型的格式說明符的語法如下:
%[argument_index$][flags][width]conversion
可選的 argument_index、flags 和 width 的定義同上。
所需的 conversion 是一個(gè)由兩字符組成的序列。第一個(gè)字符是't'或'T'。第二個(gè)字符表明所使用的格式。這些字符類似于但不完全等同于那些由 GNU date 和 POSIX strftime(3c) 定義的字符。
注意的是conversion 是一個(gè)由兩字符組成的序列。第一個(gè)字符是 t 或 T。
也就是說用conversion的時(shí)候首先必要寫一個(gè)t,然后再寫其它c(diǎn)onversion
時(shí)間類型有它自己的一套conversion,我們簡單的選擇幾個(gè)來說:
| conversion | 說明 |
|---|---|
| Y | 年份,被格式化為必要時(shí)帶前導(dǎo)零的四位數(shù)(至少),例如,0092 等于格里高利歷的92CE
|
| m | 月份,被格式化為必要時(shí)帶前導(dǎo)零的兩位數(shù),即 01 - 13。 |
| d | 日,一個(gè)月中的天數(shù),被格式化為必要時(shí)帶前導(dǎo)零兩位數(shù),即01 - 31。 |
上面三個(gè)分別表示年月日
如果我要顯示年份,我就可以%tY,顯示月份我就可以寫%tm,記得一定要帶上t
那么本篇一開始提到的那串復(fù)雜的表達(dá)式現(xiàn)在看來是不是很簡單呢:
String s2 = String.format("%1$tY-%1$tm-%1$te", new Date());
System.out.println(s2);
String.format()方法差不多講完了,仔細(xì)看JDK文檔也會(huì)慢慢了解的.
需要批量進(jìn)行格式化時(shí),考慮下DateFormat, MessageFormat, NumberFormat 把他們封裝成一個(gè)靜態(tài)工具類或許更好
畢竟調(diào)用String.format()方法是會(huì)new一個(gè)Formatter對(duì)象,雖然有GC幫忙,但是平時(shí)編程的時(shí)候還是要考慮這些因素的
盡量少的創(chuàng)建對(duì)象,節(jié)省資源。
- 常規(guī)類型的格式化
String類的format()方法用于創(chuàng)建格式化的字符串以及連接多個(gè)字符串對(duì)象。熟悉C語言的同學(xué)應(yīng)該記得C語言的sprintf()方法,兩者有類似之處。format()方法有兩種重載形式。
format(String format, Object... args) 新字符串使用本地語言環(huán)境,制定字符串格式和參數(shù)生成格式化的新字符串。
format(Locale locale, String format, Object... args) 使用指定的語言環(huán)境,制定字符串格式和參數(shù)生成格式化的字符串。
顯示不同轉(zhuǎn)換符實(shí)現(xiàn)不同數(shù)據(jù)類型到字符串的轉(zhuǎn)換,如圖所示。
| 轉(zhuǎn)換符 | 說明 | 示例 |
|---|---|---|
| %s | 字符串類型 | "mingrisoft" |
| %c | 字符類型 | 'm' |
| %b | 布爾類型 | true |
| %d | 整數(shù)類型(十進(jìn)制) | 99 |
| %x | 整數(shù)類型(十六進(jìn)制) | FF |
| %o | 整數(shù)類型(八進(jìn)制) | 77 |
| %f | 浮點(diǎn)類型 | 99.99 |
| %a | 十六進(jìn)制浮點(diǎn)類型 | FF.35AE |
| %e | 指數(shù)類型 | 9.38e+5 |
| %g | 通用浮點(diǎn)類型(f和e類型中較短的) | |
| %h | 散列碼 | |
| %% | 百分比類型 | % |
| %n | 換行符 | |
| %tx | 日期與時(shí)間類型(x代表不同的日期與時(shí)間轉(zhuǎn)換符 |
測試用例
public static void main(String[] args) {
String str=null;
str=String.format("Hi,%s", "王力");
System.out.println(str);
str=String.format("Hi,%s:%s.%s", "王南","王力","王張");
System.out.println(str);
System.out.printf("字母a的大寫是:%c %n", 'A');
System.out.printf("3>7的結(jié)果是:%b %n", 3>7);
System.out.printf("100的一半是:%d %n", 100/2);
System.out.printf("100的16進(jìn)制數(shù)是:%x %n", 100);
System.out.printf("100的8進(jìn)制數(shù)是:%o %n", 100);
System.out.printf("50元的書打8.5折扣是:%f 元%n", 50*0.85);
System.out.printf("上面價(jià)格的16進(jìn)制數(shù)是:%a %n", 50*0.85);
System.out.printf("上面價(jià)格的指數(shù)表示:%e %n", 50*0.85);
System.out.printf("上面價(jià)格的指數(shù)和浮點(diǎn)數(shù)結(jié)果的長度較短的是:%g %n", 50*0.85);
System.out.printf("上面的折扣是%d%% %n", 85);
System.out.printf("字母A的散列碼是:%h %n", 'A');
}
輸出結(jié)果
Hi,王力
Hi,王南:王力.王張
字母a的大寫是:A
3>7的結(jié)果是:false
100的一半是:50
100的16進(jìn)制數(shù)是:64
100的8進(jìn)制數(shù)是:144
50元的書打8.5折扣是:42.500000 元
上面價(jià)格的16進(jìn)制數(shù)是:0x1.54p5
上面價(jià)格的指數(shù)表示:4.250000e+01
上面價(jià)格的指數(shù)和浮點(diǎn)數(shù)結(jié)果的長度較短的是:42.5000
上面的折扣是85%
字母A的散列碼是:41
搭配轉(zhuǎn)換符的標(biāo)志,如圖所示。
| 標(biāo)志 | 說明 | 示例 | 結(jié)果 |
|---|---|---|---|
| + | 為正數(shù)或者負(fù)數(shù)添加符號(hào) | ("%+d",15) | +15 |
| ? | 左對(duì)齊 | ("%-5d",15) | 15 |
| 0 | 數(shù)字前面補(bǔ)0 | ("%04d", 99) | 0099 |
| 空格 | 在整數(shù)之前添加指定數(shù)量的空格 | ("% 4d", 99) | 99 |
| , | 以“,”對(duì)數(shù)字分組 | ("%,f", 9999.99) | 9,999.990000 |
| ( | 使用括號(hào)包含負(fù)數(shù) | ("%(f", -99.99) | (99.990000) |
| # | 如果是浮點(diǎn)數(shù)則包含小數(shù)點(diǎn),如果是16進(jìn)制或8進(jìn)制則添加0x或0 | ("%#x", 99)("%#o", 99) | 0x63 0143 |
| < | 格式化前一個(gè)轉(zhuǎn)換符所描述的參數(shù) | ("%f和%<3.2f", 99.45) | 99.450000和99.45 |
| $ | 被格式化的參數(shù)索引 | ("%1$d,%2$s", 99,"abc") | 99,abc |
測試用例
public static void main(String[] args) {
String str=null;
//$使用
str=String.format("格式參數(shù)$的使用:%1$d,%2$s", 99,"abc");
System.out.println(str);
//+使用
System.out.printf("顯示正負(fù)數(shù)的符號(hào):%+d與%d%n", 99,-99);
//補(bǔ)O使用
System.out.printf("最牛的編號(hào)是:%03d%n", 7);
//空格使用
System.out.printf("Tab鍵的效果是:% 8d%n", 7);
//.使用
System.out.printf("整數(shù)分組的效果是:%,d%n", 9989997);
//空格和小數(shù)點(diǎn)后面?zhèn)€數(shù)
System.out.printf("一本書的價(jià)格是:% 50.5f元%n", 49.8);
}
輸出結(jié)果
格式參數(shù)$的使用:99,abc
顯示正負(fù)數(shù)的符號(hào):+99與-99
最牛的編號(hào)是:007
Tab鍵的效果是: 7
整數(shù)分組的效果是:9,989,997
一本書的價(jià)格是: 49.80000元
日期和事件字符串格式化
在程序界面中經(jīng)常需要顯示時(shí)間和日期,但是其顯示的 格式經(jīng)常不盡人意,需要編寫大量的代碼經(jīng)過各種算法才得到理想的日期與時(shí)間格式。字符串格式中還有%tx轉(zhuǎn)換符沒有詳細(xì)介紹,它是專門用來格式化日期和時(shí) 間的。%tx轉(zhuǎn)換符中的x代表另外的處理日期和時(shí)間格式的轉(zhuǎn)換符,它們的組合能夠?qū)⑷掌诤蜁r(shí)間格式化成多種格式。
常見日期和時(shí)間組合的格式,如圖所示。
| 轉(zhuǎn)換符 | 說明 | 示例 |
|---|---|---|
| c | 包括全部日期和時(shí)間信息 | 星期六 十月 27 14:21:20 CST 2007 |
| F | “年-月-日”格式 | 2007-10-27 |
| D | “月/日/年”格式 | 10/27/07 |
| r | “HH:MM:SS PM”格式(12時(shí)制) | 02:25:51 下午 |
| T | “HH:MM:SS”格式(24時(shí)制) | 14:28:16 |
| R | “HH:MM”格式(24時(shí)制) | 14:28 |
測試用例
public static void main(String[] args) {
Date date=new Date();
//c的使用
System.out.printf("全部日期和時(shí)間信息:%tc%n",date);
//f的使用
System.out.printf("年-月-日格式:%tF%n",date);
//d的使用
System.out.printf("月/日/年格式:%tD%n",date);
//r的使用
System.out.printf("HH:MM:SS PM格式(12時(shí)制):%tr%n",date);
//t的使用
System.out.printf("HH:MM:SS格式(24時(shí)制):%tT%n",date);
//R的使用
System.out.printf("HH:MM格式(24時(shí)制):%tR",date);
}
輸出結(jié)果
全部日期和時(shí)間信息:星期一 九月 10 10:43:36 CST 2012
年-月-日格式:2012-09-10
月/日/年格式:09/10/12
HH:MM:SS PM格式(12時(shí)制):10:43:36 上午
HH:MM:SS格式(24時(shí)制):10:43:36
HH:MM格式(24時(shí)制):10:43
定義日期格式的轉(zhuǎn)換符可以使日期通過指定的轉(zhuǎn)換符生成新字符串。這些日期轉(zhuǎn)換符如圖所示。
public static void main(String[] args) {
Date date=new Date();
//b的使用,月份簡稱
String str=String.format(Locale.US,"英文月份簡稱:%tb",date);
System.out.println(str);
System.out.printf("本地月份簡稱:%tb%n",date);
//B的使用,月份全稱
str=String.format(Locale.US,"英文月份全稱:%tB",date);
System.out.println(str);
System.out.printf("本地月份全稱:%tB%n",date);
//a的使用,星期簡稱
str=String.format(Locale.US,"英文星期的簡稱:%ta",date);
System.out.println(str);
//A的使用,星期全稱
System.out.printf("本地星期的簡稱:%tA%n",date);
//C的使用,年前兩位
System.out.printf("年的前兩位數(shù)字(不足兩位前面補(bǔ)0):%tC%n",date);
//y的使用,年后兩位
System.out.printf("年的后兩位數(shù)字(不足兩位前面補(bǔ)0):%ty%n",date);
//j的使用,一年的天數(shù)
System.out.printf("一年中的天數(shù)(即年的第幾天):%tj%n",date);
//m的使用,月份
System.out.printf("兩位數(shù)字的月份(不足兩位前面補(bǔ)0):%tm%n",date);
//d的使用,日(二位,不夠補(bǔ)零)
System.out.printf("兩位數(shù)字的日(不足兩位前面補(bǔ)0):%td%n",date);
//e的使用,日(一位不補(bǔ)零)
System.out.printf("月份的日(前面不補(bǔ)0):%te",date);
}
輸出結(jié)果
英文月份簡稱:Sep
本地月份簡稱:九月
英文月份全稱:September
本地月份全稱:九月
英文星期的簡稱:Mon
本地星期的簡稱:星期一
年的前兩位數(shù)字(不足兩位前面補(bǔ)0):20
年的后兩位數(shù)字(不足兩位前面補(bǔ)0):12
一年中的天數(shù)(即年的第幾天):254
兩位數(shù)字的月份(不足兩位前面補(bǔ)0):09
兩位數(shù)字的日(不足兩位前面補(bǔ)0):10
月份的日(前面不補(bǔ)0):10
和日期格式轉(zhuǎn)換符相比,時(shí)間格式的轉(zhuǎn)換符要更多、更精確。它可以將時(shí)間格式化成時(shí)、分、秒甚至?xí)r毫秒等單位。格式化時(shí)間字符串的轉(zhuǎn)換符如圖所示。
| 轉(zhuǎn)換符 | 說明 | 示例 |
|---|---|---|
| H | 2位數(shù)字24時(shí)制的小時(shí)(不足2位前面補(bǔ)0) | 15 |
| I | 2位數(shù)字12時(shí)制的小時(shí)(不足2位前面補(bǔ)0) | 03 |
| k | 2位數(shù)字24時(shí)制的小時(shí)(前面不補(bǔ)0) | 15 |
| l | 2位數(shù)字12時(shí)制的小時(shí)(前面不補(bǔ)0) | 3 |
| M | 2位數(shù)字的分鐘(不足2位前面補(bǔ)0) | 03 |
| S | 2位數(shù)字的秒(不足2位前面補(bǔ)0) | 09 |
| L | 3位數(shù)字的毫秒(不足3位前面補(bǔ)0) | 015 |
| N | 9位數(shù)字的毫秒數(shù)(不足9位前面補(bǔ)0) | 562000000 |
| p | 小寫字母的上午或下午標(biāo)記 | 中:下午英:pm |
| z | 相對(duì)于GMT的RFC822時(shí)區(qū)的偏移量 | +0800 |
| Z | 時(shí)區(qū)縮寫字符串 | CST |
| s | 1970-1-1 00:00:00 到現(xiàn)在所經(jīng)過的秒數(shù) | 1193468128 |
| Q | 1970-1-1 00:00:00 到現(xiàn)在所經(jīng)過的毫秒數(shù) | 1193468128984 |
|
測試代碼
public static void main(String[] args) {
Date date = new Date();
//H的使用
System.out.printf("2位數(shù)字24時(shí)制的小時(shí)(不足2位前面補(bǔ)0):%tH%n", date);
//I的使用
System.out.printf("2位數(shù)字12時(shí)制的小時(shí)(不足2位前面補(bǔ)0):%tI%n", date);
//k的使用
System.out.printf("2位數(shù)字24時(shí)制的小時(shí)(前面不補(bǔ)0):%tk%n", date);
//l的使用
System.out.printf("2位數(shù)字12時(shí)制的小時(shí)(前面不補(bǔ)0):%tl%n", date);
//M的使用
System.out.printf("2位數(shù)字的分鐘(不足2位前面補(bǔ)0):%tM%n", date);
//S的使用
System.out.printf("2位數(shù)字的秒(不足2位前面補(bǔ)0):%tS%n", date);
//L的使用
System.out.printf("3位數(shù)字的毫秒(不足3位前面補(bǔ)0):%tL%n", date);
//N的使用
System.out.printf("9位數(shù)字的毫秒數(shù)(不足9位前面補(bǔ)0):%tN%n", date);
//p的使用
String str = String.format(Locale.US, "小寫字母的上午或下午標(biāo)記(英):%tp", date);
System.out.println(str);
System.out.printf("小寫字母的上午或下午標(biāo)記(中):%tp%n", date);
//z的使用
System.out.printf("相對(duì)于GMT的RFC822時(shí)區(qū)的偏移量:%tz%n", date);
//Z的使用
System.out.printf("時(shí)區(qū)縮寫字符串:%tZ%n", date);
//s的使用
System.out.printf("1970-1-1 00:00:00 到現(xiàn)在所經(jīng)過的秒數(shù):%ts%n", date);
//Q的使用
System.out.printf("1970-1-1 00:00:00 到現(xiàn)在所經(jīng)過的毫秒數(shù):%tQ%n", date);
}
輸出結(jié)果
2位數(shù)字24時(shí)制的小時(shí)(不足2位前面補(bǔ)0):11
2位數(shù)字12時(shí)制的小時(shí)(不足2位前面補(bǔ)0):11
2位數(shù)字24時(shí)制的小時(shí)(前面不補(bǔ)0):11
2位數(shù)字12時(shí)制的小時(shí)(前面不補(bǔ)0):11
2位數(shù)字的分鐘(不足2位前面補(bǔ)0):03
2位數(shù)字的秒(不足2位前面補(bǔ)0):52
3位數(shù)字的毫秒(不足3位前面補(bǔ)0):773
9位數(shù)字的毫秒數(shù)(不足9位前面補(bǔ)0):773000000
小寫字母的上午或下午標(biāo)記(英):am
小寫字母的上午或下午標(biāo)記(中):上午
相對(duì)于GMT的RFC822時(shí)區(qū)的偏移量:+0800
時(shí)區(qū)縮寫字符串:CST
1970-1-1 00:00:00 到現(xiàn)在所經(jīng)過的秒數(shù):1347246232
1970-1-1 00:00:00 到現(xiàn)在所經(jīng)過的毫秒數(shù):1347246232773