news 2026/5/1 7:26:59

SimpleDateFormat 为什么线程不安全

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
SimpleDateFormat 为什么线程不安全

SimpleDateFormat线程不安全的,主要原因如下:

1.内部状态可变性

// SimpleDateFormat 内部维护了可变状态 private StringBuffer format(Date date, StringBuffer toAppendTo, FieldDelegate delegate) { // 会修改内部的 Calendar 对象 calendar.setTime(date); // ... }

2.共享的 Calendar 实例

每个SimpleDateFormat对象内部都持有一个Calendar实例:

public class SimpleDateFormat extends DateFormat { protected Calendar calendar; // 共享的可变状态 public String format(Date date) { // 1. 设置时间到 calendar calendar.setTime(date); // 2. 使用 calendar 进行格式化 return format(calendar); } }

3.并发问题场景

情况1:多线程同时调用 format()

SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); // 线程A sdf.format(dateA); // 设置 calendar 为 dateA // 线程B在此刻插入 sdf.format(dateB); // 设置 calendar 为 dateB // 线程A继续格式化,但calendar已经被线程B修改

情况2:多线程同时调用 parse()

// 线程A sdf.parse("2024-01-01"); // 线程B sdf.parse("2024-02-01"); // 两者可能互相干扰,得到错误结果

4.问题复现代码

SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); ExecutorService executor = Executors.newFixedThreadPool(10); List<Future<String>> futures = new ArrayList<>(); for (int i = 0; i < 10; i++) { final int index = i; futures.add(executor.submit(() -> { Date date = new Date(System.currentTimeMillis() + index * 1000); return sdf.format(date); // 可能出现:空指针、格式错误、时间错乱 })); }

5.线程安全的替代方案

方案1:使用 ThreadLocal(推荐)

private static final ThreadLocal<SimpleDateFormat> threadLocal = ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd")); public String formatDate(Date date) { return threadLocal.get().format(date); }

方案2:每次创建新实例

public String formatDate(Date date) { return new SimpleDateFormat("yyyy-MM-dd").format(date); } // 缺点:频繁创建对象,性能较差

方案3:使用 DateTimeFormatter(Java 8+)

// DateTimeFormatter 是线程安全的 DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); // 格式化 String formatted = LocalDateTime.now().format(formatter); // 解析 LocalDateTime parsed = LocalDateTime.parse("2024-01-26 10:30:00", formatter);

方案4:使用 FastDateFormat(Apache Commons Lang)

FastDateFormat formatter = FastDateFormat.getInstance("yyyy-MM-dd"); String formatted = formatter.format(new Date()); // 线程安全

6.为什么 DateTimeFormatter 线程安全?

public final class DateTimeFormatter { // 所有字段都是 final 的 private final CompositePrinter printer; private final CompositeParser parser; private final Locale locale; // 所有方法都是纯函数,不修改内部状态 public String format(TemporalAccessor temporal) { // 不修改任何实例变量 } }

总结

  • 根本原因SimpleDateFormat内部可变状态(Calendar)在多线程下被共享修改

  • 解决方案

    1. 使用ThreadLocal包装(适合传统项目)

    2. 使用 Java 8+ 的DateTimeFormatter(推荐新项目)

    3. 使用同步锁(性能差,不推荐)

在并发环境下,永远不要共享同一个SimpleDateFormat实例

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/29 20:06:07

【Rust开发】Rust基础语法详细解析,助力你快速通关Rust

Rust基础语法解析变量与可变性Rust中变量默认不可变&#xff0c;使用let声明。可变变量需添加mut关键字&#xff1a;let x 5; // 不可变 let mut y 10; // 可变 y 15; // 允许修改常量使用const声明&#xff0c;必须标注类型&#xff1a;const MAX_POINTS: …

作者头像 李华
网站建设 2026/5/1 2:17:29

2026网络安全这趟车_你还敢上吗?

网络安全真相大揭秘&#xff1a;2026年入坑指南&#xff0c;收藏级内容&#xff0c;小白程序员必读 网络安全行业现状严峻&#xff1a;求职竞争激烈&#xff0c;企业需要的是真正理解攻防逻辑的人才而非工具使用者&#xff1b;安全部门常被视为成本中心&#xff0c;在公司预算…

作者头像 李华
网站建设 2026/5/1 3:52:40

无线控制系统如何降低轨道小车部署与维护成本?

在汽车制造、仓储物流、冶金化工等场景中&#xff0c;轨道小车是物料转运的“动脉”&#xff0c;其控制精度与运行效率直接影响生产节拍与成本。然而&#xff0c;传统有线控制模式面临布线复杂、维护成本高、移动受限等痛点&#xff0c;拖链电缆频繁弯折导致断裂、信号衰减&…

作者头像 李华
网站建设 2026/4/30 16:11:14

探索锂电池生产设备的自动化控制:从硬件到软件实现

锂电池二封机 欧姆龙NJ/NX程序NX1P2-1040DT&#xff0c; 全自动锂电池Degas机 主站NX1P2-1040DT&#xff0c;ID6142.OD6121等输入输出IO模块搭配FX5U四从站以太网通信控制实例威纶通触摸屏。 整机采用EtherCAT总线网络节点控制&#xff0c;松下A6&#xff0c;雷赛DM3E步进总线控…

作者头像 李华
网站建设 2026/4/23 17:00:27

基于单片机的红外检测及语音响应系统:车站温度检测好帮手

基于单片机的红外检测及语音响应系统 本设计是基于单片机的红外检测及语音响应系统&#xff0c;主要是为了应用于车站出入站口的温度测量。 选择STC89C52单片机作为中间的核心处理器、搭配DS18B20温度检测电路、语音播报电路、LCD显示电路。 设计的目的是&#xff1a;能够对密集…

作者头像 李华