为什么SimpleDateFormat不是线程安全的?

查看DateFormat的format(Date date)源码,我们可发现实现如下:

/**
 * 格式化日期
 */
public final String format(Date date) {
    return format(date, new StringBuffer(),
                  DontCareFieldPosition.INSTANCE).toString();
}

/**
 *  真正的格式化由此类来实现
 */
public abstract StringBuffer format(Date date, StringBuffer toAppendTo,
                                        FieldPosition fieldPosition);

继续查看SimpleDateFormat,可查看到方法相关定义如下:

public StringBuffer format(Date date, StringBuffer toAppendTo,
                           FieldPosition pos) {
    //  其实看到这里已经明白了,如此轻易地使用内部变量,能线程安全
    //  线程都对pos进行写操作,必然会影响其他线程的读操作
    pos.beginIndex = pos.endIndex = 0;
    return format(date, toAppendTo, pos.getFieldDelegate());
}

继续往下看:

 private StringBuffer format(Date date, StringBuffer toAppendTo,
                            FieldDelegate delegate) {
    // 这里已经彻底毁坏线程的安全性
    calendar.setTime(date);

    boolean useDateFormatSymbols = useDateFormatSymbols();

    for (int i = 0; i < compiledPattern.length; ) {
        int tag = compiledPattern[i] >>> 8;
        int count = compiledPattern[i++] & 0xff;
        if (count == 255) {
            count = compiledPattern[i++] << 16;
            count |= compiledPattern[i++];
        }

        switch (tag) {
        case TAG_QUOTE_ASCII_CHAR:
            toAppendTo.append((char)count);
            break;

        case TAG_QUOTE_CHARS:
            toAppendTo.append(compiledPattern, i, count);
            i += count;
            break;

        default:
            subFormat(tag, count, delegate, toAppendTo, useDateFormatSymbols);
            break;
        }
    }
    return toAppendTo;
}

结论

原以为SimpleDateFormat如此庞大的对象必然是线程安全的,尤其是只调用了其中的一个方法,还传入了日期参数,但结果却大相径庭,这种实现真是令人费解。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
SimpleDateFormat类是Java中用于格式化日期的类,它不是线程安全的。这意味着在多线程环境下同时使用一个SimpleDateFormat实例可能会导致错误的结果或者抛出异常。 要保证SimpleDateFormat线程安全,可以采用以下两种方式之一: 1. 使用ThreadLocal:可以为每个线程创建一个SimpleDateFormat实例,并将其存储在ThreadLocal对象中。这样每个线程都有自己的SimpleDateFormat实例,避免了线程间的竞争条件。 ```java private static final ThreadLocal<SimpleDateFormat> dateFormat = ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd")); public static String formatDate(Date date) { return dateFormat.get().format(date); } ``` 在上述代码中,每个线程通过`dateFormat.get()`获取自己的SimpleDateFormat实例,并调用其format方法进行日期格式化。 2. 使用局部变量:在每个需要使用SimpleDateFormat的方法内部创建一个局部变量,并在使用完毕后及时销毁。这样每个线程都有自己的SimpleDateFormat实例,避免了线程间的竞争条件。 ```java public static String formatDate(Date date) { SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd"); return dateFormat.format(date); } ``` 在上述代码中,每个方法都会创建自己的SimpleDateFormat实例,并在使用完毕后销毁,确保线程安全。 这两种方式都能够保证SimpleDateFormat线程安全性,选择哪种方式取决于具体的使用场景和需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值