【Java】String字符串格式化

摘要:
显示9个有效数字,并将四舍五入。示例:intnum=1000;Stringstr=String.format;//结果“100010001。通用类型、字符类型和数字类型的格式说明符的语法如下:%[argument_index$][flags][width][.precision]转换
一、前言

String.format() 作为文本处理工具,为我们提供强大而丰富的字符串格式化功能,为了不止步于简单调用 String.format("Hello %s", "John");,下面将笔记整理并记录下来。

其实各个语言的字符串格式化方法都是相通的,你可以在其中找到你熟悉的语言的影子,如C语言等。

二、重载方法
// 使用当前本地区域对象(Locale.getDefault())格式化字符串
String String.format(String fmt, Object... args);

// 自定义本地区域对象格式化字符串
String String.format(Locale locale, String fmt, Object... args);
三、占位符

占位符完整格式为: %[index$][标识]*[最小宽度][.精度]转换符

针对不同数据类型的格式化,占位符的格式将有所裁剪。

  • %:占位符的起始字符,若要在占位符内部使用%,则需要写成 %% 。

  • [index$]:位置索引从1开始计算,用于指定对索引相应的实参进行格式化并替换掉该占位符。

  • [标识]:用于增强格式化能力,可同时使用多个[标识],但某些标识是不能同时使用的。

  • [最小宽度]:用于设置格式化后的字符串最小长度,若使用[最小宽度]而无设置[标识],那么当字符串长度小于最小宽度时,则以左边补空格的方式凑够最小宽度。

  • [.精度]:对于浮点数类型格式化使用,设置保留小数点后多少位。

  • 转换符:用于指定格式化的样式,和限制对应入参的数据类型。

四、对字符、字符串进行格式化

占位符格式为: %[index$][标识][最小宽度]转换符

示例——将"hello"格式化为" hello"

String raw = "hello";
String str = String.format("%1$7s", raw);
// 简化
String str = String.format("%7s", raw);

示例——将"hello"格式化为"hello "

String raw = "hello";
String str = String.format("%1$-7s", raw);
// 简化
String str = String.format("%-7s", raw);

可用标识:

  • 默认:在最小宽度内右对齐,左边用空格补上;要是超过了最小宽度,那么就无效,正常显示即可。

  • -:在最小宽度内左对齐,右边用空格补上;要是超过了最小宽度,那么就无效,正常显示即可。

可用转换符:

  • s:字符串类型。

  • c:字符类型,实参必须为char或int、short等可转换为char类型的数据类型,否则抛IllegalFormatConversionException异常。

  • b:布尔类型,只要实参为非false的布尔类型,均格式化为字符串true,否则为字符串false。

  • n:平台独立的换行符(与通过 System.getProperty("line.separator") 是一样的)

五、对整数进行格式化

占位符格式为: %[index$][标识]*[最小宽度]转换符

示例——将1显示为0001

int num = 1;
String str = String.format("%04d", num);

示例——将-1000显示为(1,000)

int num = -1000;
String str = String.format("%(,d", num);

可用标识:

  • -:在最小宽度内左对齐,不可以与0标识一起使用。
  • 0:若内容长度不足最小宽度,则在左边用0来填充。
  • #:对8进制和16进制加上前导,8进制前添加一个0,16进制前添加0x。
  • +:结果总包含一个+或-号。
  • 空格:正数前加空格,负数前加-号。
  • ,:只用与十进制,每3位数字间用,分隔。
  • (:若结果为负数,则用括号括住,且不显示符号。

可用转换符:

  • b:布尔类型,只要实参为非false的布尔类型,均格式化为字符串true,否则为字符串false。
  • d:整数类型(十进制)。
  • x:整数类型(十六进制)。
  • o:整数类型(八进制)
  • n:平台独立的换行符, 也可通过System.getProperty("line.separator")获取
六、对浮点数进行格式化

占位符格式为: %[index$][标识]*[最小宽度][.精度]转换符

示例:

double num = 123.4567899;
System.out.print(String.format("%f %n", num)); // 123.456790 
System.out.print(String.format("%a %n", num)); // 0x1.edd3c0bb46929p6 
System.out.print(String.format("%g %n", num)); // 123.457

可用标识:

  • -:在最小宽度内左对齐,不可以与0标识一起使用。
  • 0:若内容长度不足最小宽度,则在左边用0来填充。
  • #:对8进制和16进制,8进制前添加一个0,16进制前添加0x。
  • +:结果总包含一个+或-号。
  • 空格:正数前加空格,负数前加-号。
  • ,:只用与十进制,每3位数字间用,分隔。
  • (:若结果为负数,则用括号括住,且不显示符号。

可用转换符:

  • b:布尔类型,只要实参为非false的布尔类型,均格式化为字符串true,否则为字符串false。
  • n:平台独立的换行符, 也可通过System.getProperty("line.separator")获取。
  • f:浮点数型(十进制)。显示9位有效数字,且会进行四舍五入。如99.99。
  • a:浮点数型(十六进制)。
  • e:指数类型。如9.38e+5。
  • g:浮点数型(比%f,%a长度短些,显示6位有效数字,且会进行四舍五入)
七、对日期时间进行格式化

占位符格式为: %[index$]t转换符

示例:

Date now = new Date();
String str = String.format("%tF", now); // 2014-10-12

可用转换符

1.日期的转换符

  • c:星期六 十月 27 14:21:20 CST 2007
  • F:2007-10-27
  • D:10/27/07
  • r:02:25:51 下午
  • T:14:28:16
  • R:14:28
  • b:月份简称
  • B:月份全称
  • a:星期简称
  • A:星期全称
  • C:年前两位(不足两位补零)
  • y:年后两位(不足两位补零)
  • j:当年的第几天
  • m:月份(不足两位补零)
  • d:日期(不足两位补零)
  • e:日期(不足两位不补零)

2.时间的转换符

  • H:24小时制的小时(不足两位补零)
  • k:24小时制的小时(不足两位不补零)
  • I:12小时制的小时(不足两位补零)
  • i:12小时制的小时(不足两位不补零)
  • M:分钟(不足两位补零)
  • S:秒(不足两位补零)
  • L:毫秒(不足三位补零)
  • N:毫秒(不足9位补零)
  • p:小写字母的上午或下午标记,如中文为“下午”,英文为pm
  • z:相对于GMT的时区偏移量,如+0800
  • Z:时区缩写,如CST
  • s:自1970-1-1 00:00:00起经过的秒数
  • Q:自1970-1-1 00:00:00起经过的豪秒
八、其他转换符

<,用于格式化前一个转换符所描述的参数。

示例:

int num = 1000;
String str = String.format("%d %<,d", num);
// 结果"1000 1,000

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

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

  • 可选的 argument_index 是一个十进制整数,用于表明参数在参数列表中的位置。第一个参数由 "1(" 引用,第二个参数由 "2)" 引用,依此类推。
  • 可选的 flags 是修改输出格式的字符集。有效标志的集合取决于转换类型。
  • 可选 width 是一个非负十进制整数,表明要向输出中写入的最少字符数。
  • 可选 precision 是一个非负十进制整数,通常用来限制字符数。特定行为取决于转换类型。
  • 必须的 conversion 是一个表明应该如何格式化参数的字符。给定参数的有效转换集合取决于参数的数据类型。

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

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

  • 可选的 argument_indexflagswidth 的定义同上。
  • 必须的 conversion 是一个由两字符组成的序列。第一个字符是 't' 或 'T'。第二个字符表明所使用的格式。这些字符类似于但不完全等同于那些由 GNU date 和 POSIX strftime(3c) 定义的字符。

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

%[flags][width]conversion

  • 可选 flagswidth 的定义同上。
  • 必须的 conversion 是一个表明要在输出中所插内容的字符。

转换

转换可分为以下几类:

  1. 常规:可应用于任何参数类型
  2. 字符:可应用于表示 Unicode 字符的基本类型:char、Character、byte、Byte、short 和 Short。
    当 Character.isValidCodePoint(int) 返回 true 时,可将此转换应用于 int 和 Integer 类型
  3. 数值:
    1. 整数:可应用于 Java 的整数类型:byte、Byte、short、Short、int、Integer、long、Long 和 BigInteger
    2. 浮点:可用于 Java 的浮点类型:float、Float、double、Double 和 BigDecimal
  4. 日期/时间:可应用于 Java 的、能够对日期或时间进行编码的类型:long、Long、Calendar 和 Date。
  5. 百分比:产生字面值 '%' ('u0025')
  6. 行分隔符:产生特定于平台的行分隔符

格式化表达式:%[零个或多个标志][最小字段宽度][精度][修改符]格式码

注意:[]方括号表示可选参数

常规类型的格式化

String类的format()方法用于创建格式化的字符串以及连接多个字符串对象。熟悉C语言的同学应该记得C语言的sprintf()方法,两者有类似之处。

format()方法有两种重载形式。

format(String format, Object... args):新字符串使用本地语言环境,制定字符串格式和参数生成格式化的新字符串。

format(Locale locale, String format, Object... args):使用指定的语言环境,制定字符串格式和参数生成格式化的字符串。

转换符

显示不同转换符实现不同数据类型到字符串的转换,如图所示。

转换符说明示例
%s字符串类型"mingrisoft"
%c字符类型'm'
%b布尔类型true
%d整数类型(十进制)99
%x整数类型(十六进制)FF
%o整数类型(八进制)77
%f浮点类型99.99
%a十六进制浮点类型FF.35AE
%e指数类型9.38e+5
%g通用浮点类型(f和e类型中较短的)
%h散列码
%%百分比类型
%n换行符
%tx日期与时间类型(x代表不同的日期与时间转换符)

测试用例

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的结果是:%b %n", 3>7);
    System.out.printf("100的一半是:%d %n", 100/2);
    System.out.printf("100的16进制数是:%x %n", 100);
    System.out.printf("100的8进制数是:%o %n", 100);
    System.out.printf("50元的书打8.5折扣是:%f 元%n", 50*0.85);
    System.out.printf("上面价格的16进制数是:%a %n", 50*0.85);
    System.out.printf("上面价格的指数表示:%e %n", 50*0.85);
    System.out.printf("上面价格的指数和浮点数结果的长度较短的是:%g %n", 50*0.85);
    System.out.printf("上面的折扣是%d%% %n", 85);
    System.out.printf("字母A的散列码是:%h %n", 'A');
}

输出结果

Hi,王力
Hi,王南:王力.王张
字母a的大写是:A 
3>7的结果是:false 
100的一半是:50 
100的16进制数是:64 
100的8进制数是:144 
50元的书打8.5折扣是:42.500000 元
上面价格的16进制数是:0x1.54p5 
上面价格的指数表示:4.250000e+01 
上面价格的指数和浮点数结果的长度较短的是:42.5000 
上面的折扣是85% 
字母A的散列码是:41 

标识符

搭配转换符的标志,如图所示。

标志说明示例结果
+为正数或者负数添加符号("%+d",15)+15
左对齐("%-5d",15)15
0数字前面补0("%04d", 99)0099
空格在整数之前添加指定数量的空格("% 4d", 99)99
,以“,”对数字分组("%,f", 9999.99)9,999.990000
(使用括号包含负数("%(f", -99.99)(99.990000)
#如果是浮点数则包含小数点,如果是16进制或8进制则添加0x或0("%#x", 99)
("%#o", 99)
0x63
0143
<格式化前一个转换符所描述的参数("%f和%<3.2f", 99.45)99.450000和99.45
$被格式化的参数索引("%1$d,%2$s", 99,"abc")99,abc

测试用例

public static void main(String[] args) {
    String str = null;

    //$使用
    str=String.format("格式参数$的使用:%1$d,%2$s", 99,"abc");
    System.out.println(str);

    //+使用
    System.out.printf("显示正负数的符号:%+d与%d%n", 99,-99);

    //补O使用
    System.out.printf("最牛的编号是:%03d%n", 7);

    //空格使用
    System.out.printf("Tab键的效果是:% 8d%n", 7);

    //.使用
    System.out.printf("整数分组的效果是:%,d%n", 9989997);

    //空格和小数点后面个数
    System.out.printf("一本书的价格是:% 50.5f元%n", 49.8);
}  

输出结果

格式参数$的使用:99,abc
显示正负数的符号:+99与-99
最牛的编号是:007
Tab键的效果是:       7  
整数分组的效果是:9,989,997
一本书的价格是:                                          49.80000元

日期和事件字符串格式化

在程序界面中经常需要显示时间和日期,但是其显示的格式经常不尽人意,需要编写大量的代码经过各种算法才得到理想的日期与时间格式。字符串格式中还有%tx转换符没有详细介绍,它是专门用来格式化日期和时间的。%tx转换符中的x代表另外的处理日期和时间格式的转换符,它们的组合能够将日期和时间格式化成多种格式。

常见日期和时间组合的格式,如图所示。

转换符说明示例
c包括全部日期和时间信息星期六 十月 27 14:21:20 CST 2007
F“年-月-日”格式2007-10-27
D“月/日/年”格式10/27/07
r“HH:MM:SS PM”格式(12时制)02:25:51 下午
T“HH:MM:SS”格式(24时制)14:28:16
R“HH:MM”格式(24时制)14:28

测试用例

public static void main(String[] args) {
    Date date = new Date();

    //c的使用
    System.out.printf("全部日期和时间信息:%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时制):%tr%n",date);

    //t的使用
    System.out.printf("HH:MM:SS格式(24时制):%tT%n",date);

    //R的使用
    System.out.printf("HH:MM格式(24时制):%tR",date);
}  

输出结果

全部日期和时间信息:星期一 九月 10 10:43:36 CST 2012
年-月-日格式:2012-09-10
月/日/年格式:09/10/12
HH:MM:SS PM格式(12时制):10:43:36 上午
HH:MM:SS格式(24时制):10:43:36
HH:MM格式(24时制):10:43

定义日期格式的转换符可以使日期通过指定的转换符生成新字符串。这些日期转换符如图所示。

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("年的前两位数字(不足两位前面补0):%tC%n",date);

    //y的使用,年后两位
    System.out.printf("年的后两位数字(不足两位前面补0):%ty%n",date);

    //j的使用,一年的天数
    System.out.printf("一年中的天数(即年的第几天):%tj%n",date);

    //m的使用,月份
    System.out.printf("两位数字的月份(不足两位前面补0):%tm%n",date);

    //d的使用,日(二位,不够补零)
    System.out.printf("两位数字的日(不足两位前面补0):%td%n",date);

    //e的使用,日(一位不补零)
    System.out.printf("月份的日(前面不补0):%te",date);
}  

输出结果

英文月份简称:Sep
本地月份简称:九月
英文月份全称:September
本地月份全称:九月
英文星期的简称:Mon
本地星期的简称:星期一
年的前两位数字(不足两位前面补0):20
年的后两位数字(不足两位前面补0):12
一年中的天数(即年的第几天):254
两位数字的月份(不足两位前面补0):09
两位数字的日(不足两位前面补0):10
月份的日(前面不补0):10

和日期格式转换符相比,时间格式的转换符要更多、更精确。它可以将时间格式化成时、分、秒甚至时毫秒等单位。格式化时间字符串的转换符如图所示。

转换符说明示例
H2位数字24时制的小时(不足2位前面补0)15
I2位数字12时制的小时(不足2位前面补0)03
k2位数字24时制的小时(前面不补0)15
l2位数字12时制的小时(前面不补0)3
M2位数字的分钟(不足2位前面补0)03
S2位数字的秒(不足2位前面补0)09
L3位数字的毫秒(不足3位前面补0)015
N9位数字的毫秒数(不足9位前面补0)562000000
p小写字母的上午或下午标记中:下午
英:pm
z相对于GMT的RFC822时区的偏移量+0800
Z时区缩写字符串CST
s1970-1-1 00:00:00 到现在所经过的秒数1193468128
Q1970-1-1 00:00:00 到现在所经过的毫秒数1193468128984

测试代码

public static void main(String[] args) {
    Date date = new Date();

    //H的使用
    System.out.printf("2位数字24时制的小时(不足2位前面补0):%tH%n", date);

    //I的使用
    System.out.printf("2位数字12时制的小时(不足2位前面补0):%tI%n", date);

    //k的使用
    System.out.printf("2位数字24时制的小时(前面不补0):%tk%n", date);

    //l的使用
    System.out.printf("2位数字12时制的小时(前面不补0):%tl%n", date);

    //M的使用
    System.out.printf("2位数字的分钟(不足2位前面补0):%tM%n", date);

    //S的使用
    System.out.printf("2位数字的秒(不足2位前面补0):%tS%n", date);

    //L的使用
    System.out.printf("3位数字的毫秒(不足3位前面补0):%tL%n", date);

    //N的使用
    System.out.printf("9位数字的毫秒数(不足9位前面补0):%tN%n", date);

    //p的使用
    String str = String.format(Locale.US, "小写字母的上午或下午标记(英):%tp", date);
    System.out.println(str);
    System.out.printf("小写字母的上午或下午标记(中):%tp%n", date);

    //z的使用
    System.out.printf("相对于GMT的RFC822时区的偏移量:%tz%n", date);

    //Z的使用
    System.out.printf("时区缩写字符串:%tZ%n", date);

    //s的使用
    System.out.printf("1970-1-1 00:00:00 到现在所经过的秒数:%ts%n", date);

    //Q的使用
    System.out.printf("1970-1-1 00:00:00 到现在所经过的毫秒数:%tQ%n", date);
}

输出结果

2位数字24时制的小时(不足2位前面补0):11
2位数字12时制的小时(不足2位前面补0):11
2位数字24时制的小时(前面不补0):11
2位数字12时制的小时(前面不补0):11
2位数字的分钟(不足2位前面补0):03
2位数字的秒(不足2位前面补0):52
3位数字的毫秒(不足3位前面补0):773
9位数字的毫秒数(不足9位前面补0):773000000
小写字母的上午或下午标记(英):am
小写字母的上午或下午标记(中):上午
相对于GMT的RFC822时区的偏移量:+0800
时区缩写字符串:CST
1970-1-1 00:00:00 到现在所经过的秒数:1347246232
1970-1-1 00:00:00 到现在所经过的毫秒数:1347246232773
总结

详情可以查阅jdk5之后的官方文档中的java.util.Formatter类,有相关说明,本文的所有演示都取自此处。

免责声明:文章转载自《【Java】String字符串格式化》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇【IOS】IOS8 TabBarItem设置自定义图片问题C# 程序打包Release版本下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Apache Shiro(六)-基于URL配置权限

数据库 先准备数据库啦。 1 DROP DATABASE IF EXISTS shiro; 2 CREATE DATABASE shiro DEFAULT CHARACTER SET utf8; 3 USE shiro; 4 5 drop table if exists user; 6 drop table if exis...

【改进】用Log4net建立日志记录

上一篇随笔中只使用了普通的文件读写来进行日志的写入,正如很多朋友说的,频繁的对文件进行读写会造成很多的问题,代码缺少边界控制和操作控制,没有对资源进行管理,是非常典型的bad code。 然后经过前辈们的提点,今天使用了Log4net进行日志的写入,发现非常的便捷,同时也集成了对于日志的控制,减少因为日志写入而发生的一系列不该有的错误。 网上也有很多教程,...

Kotlin入门(14)继承的那些事儿

上一篇文章介绍了类对成员的声明方式与使用过程,从而初步了解了类的成员及其运用。不过早在《Kotlin入门(12)类的概貌与构造》中,提到MainActivity继承自AppCompatActivity,而Kotlin对于类继承的写法是“class MainActivity : AppCompatActivity() {}”,这跟Java对比有明显差异,那么...

android之Parcel机制学习

Parcel源码的分析,可以参考曹文斌的探索Android中的Parcel机制(上)。 本质上把Parcel当成一个Serialize,不同在于它是在内存中完成的序列化和反序列化,利用的是连续的内存空间,因此会更加高效。 在Android中,最常见的使用Parcel类的情况是:在Activity间传递数据。在Activity间通过Intent传递数据的时候...

JVM(七)JVM常量池详解

一、Class常量池解析   定义:Class常量池可以理解为是Class文件中的资源仓库。   内容:Class文件中除了包含类的版本、字段、方法、接口等描述信息外, 还有一项信息就是常量池,用于存放编译期生成的各种字面量和符号引用。   我们可以通过一个命令来查看我们字节码文件的内容: 字面量   定义:字面量就是指由字母、数字等构成的字符串或者数值...

golang的传值调用和传引用调用

传值还是传引用 调用函数时, 传入的参数的 传值 还是 传引用, 几乎是每种编程语言都会关注的问题. 最近在使用 golang 的时候, 由于 传值 和 传引用 的方式没有弄清楚, 导致了 BUG. 经过深入的尝试, 终于弄明白了 golang 的 传值 的 传引用, 尝试过程记录如下, 供大家参考! golang 本质上都是传值方式调用 严格来说, go...