专栏首页Spring专栏还在用SimpleDateFormat格式化时间?小心经理锤你
原创

还在用SimpleDateFormat格式化时间?小心经理锤你

还在用SimpleDateFormat格式化时间?小心经理锤你

SimpleDateFormat

场景

本来开开心心的周末时光,线上突然就疯狂报错,以为程序炸了,截停日志,发现是就是类似下述一段错误

java.lang.NumberFormatException: For input string: ".202006E.202006E44"
	at sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:2043)
	at sun.misc.FloatingDecimal.parseDouble(FloatingDecimal.java:110)
	at java.lang.Double.parseDouble(Double.java:538)
	at java.text.DigitList.getDouble(DigitList.java:169)
	at java.text.DecimalFormat.parse(DecimalFormat.java:2089)
	at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1869)
	at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)
	at java.text.DateFormat.parse(DateFormat.java:364)
	at com.github.springtools.SimpleDateFormatTest.lambda$null$0(Xxxxxxx.java:2020)
	at java.lang.Thread.run(Thread.java:748)

定位到错误处,发现是一个时间格式化(SimpleDateFormat)的异常,一个时间格式化怎么会导致这种错误,还使得接口不能正常调用

测试

拉出来,使用模拟接口多线程的环境,单独进行测试.....

package com.github.springtools;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.stream.IntStream;

public class SimpleDateFormatTest {
    public static final SimpleDateFormat FORMAT = new SimpleDateFormat("yyyy-MM-dd");

    public static void main(String[] args) throws ParseException {
        System.out.println(FORMAT.parse("2020-06-06"));
        System.out.println("--------单个调用结束--------");
        System.out.println("--------多线程调用开始--------");
        
        IntStream.rangeClosed(0, 10)
                .forEach(i -> new Thread(() -> {
                    try {
                        System.out.println(FORMAT.parse("2020-06-06"));
                    } catch (ParseException e) {
                        e.printStackTrace();
                    }
                }).start());
    }
}

输出

Sat Jun 06 00:00:00 CST 2020
--------单个调用结束--------
--------多线程调用开始--------
Sat Jun 06 00:00:00 CST 2020
Sat Jun 06 00:00:00 CST 2020
Exception in thread "Thread-7" Exception in thread "Thread-8" java.lang.NumberFormatException: For input string: ".202006E.202006E44"
	at sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:2043)
	at sun.misc.FloatingDecimal.parseDouble(FloatingDecimal.java:110)
	at java.lang.Double.parseDouble(Double.java:538)

罪魁祸手浮出水面,就是SimpleDateFormat的锅

有趣

线程不安全,去找Java文档里的SimpleDateFormat: https://docs.oracle.com/javase/7/docs/api/java/text/SimpleDateFormat.html

Date formats are not synchronized. It is recommended to create separate format instances for each thread. If multiple threads access a format concurrently, it must be synchronized externally.日期格式不同步。建议为每个线程创建单独的格式实例。如果多个线程同时访问一种格式,则必须在外部进行同步。

解决方法

使用ThreadLocal

// ThreadLocal
public static final ThreadLocal<SimpleDateFormat> THREADLOCAL_FORMAT = ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd"));
IntStream.rangeClosed(0, 5)
        .forEach(i -> new Thread(() -> {
            try {
                System.out.println("ThreadLocal:" + THREADLOCAL_FORMAT.get().parse("2020-06-06"));
            } catch (ParseException e) {
                e.printStackTrace();
            }
        }).start());

使用Java 8中的时间处理

public static final DateTimeFormatter JAVA8_FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd");
IntStream.rangeClosed(0, 5)
        .forEach(i -> new Thread(() -> {
            try {
                System.out.println("JAVA8_FORMATTER:" + JAVA8_FORMATTER.parse("2020-06-06"));
            } catch (ParseException e) {
                e.printStackTrace();
            }
        }).start());

总结

使用多线程的时候,一定要考虑到其调用到的实例变量,Java8中时间格式化DateTimeFormatter是用final修饰的,不可变类,所以是线程安全的,或者在线程中调用ThreadLocal也是可以的

原创声明,本文系作者授权云+社区发表,未经许可,不得转载。

如有侵权,请联系 yunjia_community@tencent.com 删除。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • IDEA提高开发效率的7个插件

    一般我们为了新建多表连接后映射的 ResultMap ,耗费不少时间,那么我们就来试一试这个多行编辑

    拾荒者的笔记
  • MongoDB快速入门指南与docker-compose快体验

    拾荒者的笔记
  • 一文整懂 Java 中静态代码块 / 初始块 / 构造方法的执行顺序

    “ 相信,刷过面试题应该都碰到过很多题,关于类继承后 Java 中静态代码块 / 初始块 / 构造方法的执行顺序问题,每每记一下又忘了,那么,今天来用不多的时间...

    拾荒者的笔记
  • RestAssured接口系列|环境搭建

    http://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.h...

    测试邦
  • Serializable接口心得总结

    可以看到该类的内部实现完全为空,在Java IO体系中仅起一个标记的作用。那么这个标记具体是如何发挥作用的呢?我们测试一下:

    Remember_Ray
  • tomcat shutdown.sh结束不了,Could not contact localhost:8005

    使用./shutdown.sh关闭Tomcat,有时会关闭成功,有时会出现关闭错误;

    拓荒者
  • Serializable接口心得总结

    可以看到该类的内部实现完全为空,在Java IO体系中仅起一个标记的作用。那么这个标记具体是如何发挥作用的呢?我们测试一下:

    Remember_Ray
  • 反射基础之Field

    java.lang.reflect.Field 类的方法可以查询字段的信息。比如:名字,类型,修饰符和注解。同样也有方法可以动态访问和修改字段的值。

    代码拾遗
  • linux tomcat 无法关闭 :8005端口未启动

    版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/qq_25283709/article/details/550...

    拓荒者
  • 线程池的execute方法和submit方法有什么区别?

    concurrent包里的ExecutorService,是一个接口,继承的是Executor,而Executor里只有一个方法。

    xjjdog

扫码关注云+社区

领取腾讯云代金券