java simpledateformat string_Java SimpleDateFormat 没那么简单

原标题:Java SimpleDateFormat 没那么简单

dzone.com/articles/java-simpledateformat-is-not-simple

Java 日期格式化与解析是一项日常(痛苦的)任务,每天都让我们头痛不已。

通常使用SimpleDateFormat,下面是一个常见的日期工具类。

importjava.text.ParseException;

importjava.text.SimpleDateFormat;

importjava.util.Date;

publicfinal classDateUtils {

publicstaticfinal SimpleDateFormat SIMPLE_DATE_FORMAT = newSimpleDateFormat( "yyyy-MM-dd");

privateDateUtils{}

publicstaticDate parse(String target){

try{

returnSIMPLE_DATE_FORMAT.parse(target);

} catch(ParseException e) {

e.printStackTrace;

}

returnnull;

}

publicstaticString format(Date target){

returnSIMPLE_DATE_FORMAT.format(target);

}

}

感觉可以按预期运行并输出结果吗?让我们试一下。

privatestaticvoidtestSimpleDateFormatInSingleThread{

final String source = "2019-01-11";

System.out.println(DateUtils.parse(source));

}

// Fri Jan 11 00:00:00 IST 2019

运行成功,让我们加上多线程。

privatestaticvoidtestSimpleDateFormatWithThreads{

ExecutorService executorService = Executors.newFixedThreadPool( 10);

final String source = "2019-01-11";

System.out.println( ":: parsing date string ::");

IntStream.rangeClosed( 0, 20)

.forEach((i) -> executorService.submit( -> System.out.println(DateUtils.parse(source))));

executorService.shutdown;

}

下面是我得到的运行结果。

:: parsing date string::

... omitted

Fri Jan 1100: 00: 00IST 2019

Sat Jul 1100: 00: 00IST 2111

Fri Jan 1100: 00: 00IST 2019

... omitted

结果看上去很奇怪,对吧?这是大多数人用 Java 格式化日期时常犯的一个错误。为什么会有这种奇怪的结果?因为没有考虑到到线程安全。以下是 Java 文档有关SimpleDateFormat的描述:

>

“日期格式是非同步的。

建议为每个线程创建单独的日期格式化实例。

如果多个线程并发访问某个格式化实例,则必须保证外部调用同步性。“

>

提示:使用实例变量时,应该每次检查这个类是不是线程安全。

正如文档中提到的那样,可以为每个线程设置不同实例来解决这个问题。如果要共享实例,该如何实现?

1. ThreadLocal

可以使用ThreadLocal解决。Threadlocal 的 get 方法会给当前线程提供正确的值。

importjava.text.DateFormat;

importjava.text.ParseException;

importjava.text.SimpleDateFormat;

importjava.util.Date;

publicfinal classDateUtilsThreadLocal {

publicstaticfinal ThreadLocal SIMPLE_DATE_FORMAT = ThreadLocal

.withInitial( -> newSimpleDateFormat( "yyyy-MM-dd"));

privateDateUtilsThreadLocal{}

publicstaticDate parse(String target){

try{

return((DateFormat) SIMPLE_DATE_FORMAT.get).parse(target);

} catch(ParseException e) {

e.printStackTrace;

}

returnnull;

}

publicstaticString format(Date target){

return((DateFormat) SIMPLE_DATE_FORMAT.get).format(target);

}

}

译注:实际运行时需要加上强制类型转换,否则报告编译错误。

2. Java 8 线程安全的时间日期 API

Java8 引入了新的日期时间 API,SimpleDateFormat有了更好的替代者。如果继续坚持使用 SimpleDateFormat 可以配合 ThreadLocal 一起使用。但既然已经有了更好的选择,还是考虑用新的 API。

Java 8 提供了几个线程安全的日期类,Java 文档中这么描述:

“这个类是具有不可变和线程安全的特点。”

非常值得学习这些类的用法,包括 DateTimeFormatter、OffsetDateTime、ZonedDateTime、LocalDateTime、LocalDate 和 LocalTime。

使用新 API 后的代码:

importjava.time.LocalDate;

importjava.time.format.DateTimeFormatter;

publicclassDateUtilsJava8 {

publicstaticfinal DateTimeFormatter DATE_TIME_FORMATTER = DateTimeFormatter.ofPattern( "yyyy-MM-dd");

privateDateUtilsJava8{}

publicstaticLocalDate parse(String target){

returnLocalDate.parse(target, DATE_TIME_FORMATTER);

}

publicstaticString format(LocalDate target){

returntarget.format(DATE_TIME_FORMATTER);

}

}

总结

Java 8 提供的不可变时间是一种解决多日期类线程问题的最佳实践。不可变类本质上是线程安全的,应当尽可能使用。

编程快乐!

(点击标题可跳转阅读)返回搜狐,查看更多

责任编辑:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值