多线程环境下安全地使用 SimpleDateFormat的常见方法
文章目录
- 1. 使用局部变量(每个线程独立一个实例)
- 2. 使用 ThreadLocal<SimpleDateFormat>
- 3. 使用 DateTimeFormatter(Java 8 及以上)
- 4. 使用 DateFormat 子类(如 FastDateFormat)
- 5. 使用 synchronized 或其他锁(不推荐)
因为,SimpleDateFormat类的内部有一个Calendar对象引用,这个对象主要用来储存和这个
SimpleDateFormat相关的日期信息。
当我们把SimpleDateFormat作为多个线程的共享资源来使用的时候,那就意味着多个线程之间会共享这个
SimpleDateFormat里面的Calendar引用。如果多个线程同时于操作这个Calendar对象的情况下,就会出现数据脏
读的现象,从而导致一些不可预料的错误。
那么,保证SimpleDateFormat线程安全呢
1)、可以把SimpleDateFormat定义成非全局使用的局部变量,这样每个线程调用的时候都创建一个新的实例。
2)、可以使用ThreadLocal,把SimpleDateFormat变成一个线程私有的对象。
3)、定义SimpleDateFormat的时候,加上同步锁,这样就能够保证在同一时刻只允许一个线程操作
4)、使用Java 8的新特性,在Java8中引入了一些线程安全的日期操作API,比如LocalDateTimer、
DateTimeFormatter 等等。
SimpleDateFormat oldFormatter = new SimpleDateFormat("yyyy/MM/dd");
Date date1 = new Date();
System.out.println(oldFormatter.format(date1));
// Java 8
DateTimeFormatter newFormatter = DateTimeFormatter.ofPattern("yyyy/MM/dd");
LocalDate date2 = LocalDate.now();
System.out.println(date2.format(newFormatter));static final ThreadLocal<SimpleDateFormat> SIMPLE_DATE_FORMAT_LOCAL = ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")
);
SimpleDateFormat 在 Java 中是一个线程不安全的类,因此在多线程环境下直接共享同一个 SimpleDateFormat 实例可能导致意外的结果或错误。例如,多个线程共享一个 SimpleDateFormat 实例时,可能会互相覆盖彼此的格式化结果,产生不可预测的行为。
最优解:使用 ThreadLocal 或 DateTimeFormatter(Java 8 及以上)是最佳选择。
不推荐的方案:synchronized 锁定 SimpleDateFormat 实例,性能开销较大。
推荐的替代品:如果可以使用 Java 8 及以上版本,优先使用 DateTimeFormatter,因为它是线程安全且现代化的日期时间格式化工具。
选择适合您项目需求的方案,以确保在多线程环境下安全有效地使用日期格式化。
为了在多线程环境下安全地使用 SimpleDateFormat,有几个常见的解决方法:
1. 使用局部变量(每个线程独立一个实例)
最简单的方法是让每个线程都有自己独立的 SimpleDateFormat 实例。这可以通过将 SimpleDateFormat 定义为方法内的局部变量来实现,这样每个线程在调用方法时都会创建并使用自己的实例。
public String formatDate(Date date) {SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");return sdf.format(date);
}
这样每个线程都使用自己独立的 SimpleDateFormat 实例,避免了线程安全问题。
2. 使用 ThreadLocal
如果需要在多个方法或类中使用 SimpleDateFormat,可以考虑使用 ThreadLocal 来确保每个线程有自己独立的 SimpleDateFormat 实例。ThreadLocal 可以为每个线程提供一个独立的变量副本,从而避免共享 SimpleDateFormat 实例导致的线程安全问题。
public class DateFormatUtil {// 使用 ThreadLocal 来为每个线程提供独立的 SimpleDateFormat 实例private static final ThreadLocal<SimpleDateFormat> threadLocalDateFormat = ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));public static String formatDate(Date date) {SimpleDateFormat sdf = threadLocalDateFormat.get();return sdf.format(date);}
}
解释:
ThreadLocal.withInitial() 创建了一个 ThreadLocal 实例,并初始化每个线程独立的 SimpleDateFormat。
每个线程调用 threadLocalDateFormat.get() 时都会获取该线程独有的 SimpleDateFormat 实例,从而避免了多线程间的共享问题。
3. 使用 DateTimeFormatter(Java 8 及以上)
从 Java 8 开始,java.time 包中的 DateTimeFormatter 提供了线程安全的日期时间格式化工具。DateTimeFormatter 是不可变的,并且设计时已经考虑到线程安全问题,因此推荐在 Java 8 及以上版本使用 DateTimeFormatter 来代替 SimpleDateFormat。
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
public class DateFormatUtil {
// 使用线程安全的 DateTimeFormatter
private static final DateTimeFormatter formatter = DateTimeFormatter.ofPattern(“yyyy-MM-dd HH:mm:ss”);
public static String formatDate(LocalDateTime dateTime) {return dateTime.format(formatter);
}
}
解释:
DateTimeFormatter 是不可变的,可以在多个线程中共享而不产生线程安全问题。
该类与 LocalDateTime 等类配合使用,支持更丰富的日期时间处理。
4. 使用 DateFormat 子类(如 FastDateFormat)
如果你使用 Apache Commons Lang 库,可以使用 FastDateFormat 替代 SimpleDateFormat。FastDateFormat 是线程安全的,设计时就考虑到了性能和线程安全问题。
import org.apache.commons.lang3.time.FastDateFormat;public class DateFormatUtil {// 使用 FastDateFormat 来代替 SimpleDateFormatprivate static final FastDateFormat fastDateFormat = FastDateFormat.getInstance("yyyy-MM-dd HH:mm:ss");public static String formatDate(Date date) {return fastDateFormat.format(date);}
}
解释:
FastDateFormat 提供了与 SimpleDateFormat 类似的功能,但它是线程安全的,可以在多线程环境中安全使用。
5. 使用 synchronized 或其他锁(不推荐)
另一种方法是通过对 SimpleDateFormat 实例加锁来保证线程安全。尽管这种方法可以确保线程安全,但会导致性能瓶颈,因此不推荐这种方法,特别是在高并发的情况下。
public class DateFormatUtil {private static final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");public synchronized static String formatDate(Date date) {return sdf.format(date);}
}
解释:
使用 synchronized 确保每次只有一个线程可以访问 formatDate 方法。但这种方法的性能较差,因为每次格式化都会等待锁。
相关文章:
多线程环境下安全地使用 SimpleDateFormat的常见方法
文章目录 1. 使用局部变量(每个线程独立一个实例)2. 使用 ThreadLocal<SimpleDateFormat>3. 使用 DateTimeFormatter(Java 8 及以上)4. 使用 DateFormat 子类(如 FastDateFormat)5. 使用 synchronize…...
easyexcel实现自定义的策略类, 最后追加错误提示列, 自适应列宽,自动合并重复单元格, 美化表头
easyexcel实现自定义的策略类, 最后追加错误提示列, 自适应列宽,自动合并重复单元格, 美化表头 原版表头和表体字体美化自动拼接错误提示列自适应宽度自动合并单元格使用Easyexcel使用poi导出 在后台管理开发的工作中,离不开的就是导出excel了. 如果是简单的导出, 直接easyexce…...
ANDROIDWORLD: A Dynamic Benchmarking Environment for Autonomous Agents论文学习
这个任务是基于androidenv的。这个环境之前学过,是一个用来进行强化学习的线上环境。而这篇文章的工作就是要给一些任务加上中间的奖励信号。这种训练环境的优点就是动态,与静态的数据集(比如说我自己的工作)不同,因此…...
Docker 常用命令详解(详细版)
Docker 是一个开源的容器化平台,它使得开发人员可以打包应用程序及其所有依赖项,并在任何环境中运行。Docker 提供了简单而强大的命令行工具来管理容器、镜像、网络等。本文将详细介绍 Docker 的常用命令及其使用方法。 1. 安装 Docker 在使用 Docker …...
【网络安全 | 甲方安全建设】分布式系统、Redis分布式锁及Redisson看门狗机制
未经许可,不得转载。 文章目录 分布式系统分布式系统的核心特性分布式系统的典型架构分布式锁概念Redis 分布式锁原理互斥性锁释放锁的唯一性具体实现Redisson分布式锁分布式系统 分布式系统是一种由多台计算机(节点)组成的系统,这些节点通过网络相互连接并协同工作,共同…...
「QT」几何数据类 之 QLineF 浮点型直线类
✨博客主页何曾参静谧的博客📌文章专栏「QT」QT5程序设计📚全部专栏「VS」Visual Studio「C/C」C/C程序设计「UG/NX」BlockUI集合「Win」Windows程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「PK」Parasolid…...
Treeland 技术揭秘,如何使得 DDE 纵享丝滑?
近日,deepin(深度)社区亮相COSCon24 第九届中国开源年会开源市集,且社区资深桌面研发工程师张丁元为大家来了《Treeland,DDE进步的阶梯》技术分享。 就着这个机会,今天就让我们一起来聊聊如何在追求华丽动…...
快速了解SpringBoot 统一功能处理
拦截器 什么是拦截器: 拦截器是Spring框架提供的重要功能之一,主要进行拦截用户请求,在指定方法前后,根据业务需求,执行预先设定的代码。 也就是说,允许开发⼈员提前预定义⼀些逻辑,在⽤⼾的请求响应前后执⾏.也可以…...
C++区分数组的引用和引用的数组
void f(int (&arr)[10]) {//正确} void f1(int &arr[10]) {//不允许使用引用的数组} [](数组下标运算符)的优先级高于&(取地址运算符)。所有表达式&arr[i]等价于&(arr[i]) 引用的数组 一个包含引用的数组&…...
【harbor】离线安装2.9.0-arm64架构服务制作和升级部署
harbor官网地址:Harbor 参考文档可以看这里:部署 harbor 2.10.1 arm64 - 简书。 前提环境准备: 安装docker 和 docker-compose 先拉arm64架构的harbor相关镜像 docker pull --platformlinux/arm64 ghcr.io/octohelm/harbor/harbor-regist…...
ESLint 使用教程(五):ESLint 和 Prettier 的结合使用与冲突解决
前言 在现代前端开发中,代码质量与代码风格的统一是两个非常重要的方面。良好的代码质量能减少 bug 的产生,而统一的代码风格则能提高团队协作的效率。为了实现这两个目标,我们通常会使用一些工具。 为了保证代码的可读性和维护性࿰…...
uniApp之uni-file-picker使用踩坑
标题党~也不算坑吧 就是初体验 上传是需要存储一下子的,我以为uniApp是自己免费开的服务给大家中转使用,就没管这个事,但是官网是这么说的: 就我是怎么发现的,使用了一段时间后,上传的图片都裂了ÿ…...
【C语言】。末尼
用js写一个冒泡排序 当然可以。在JavaScript中,你可以使用递归的方式来实现冒泡排序算法。这是一个简单的示例: function bubbleSort(arr) { let len arr.length; if(len < 1){ return arr; // 如果数组只有一个元素或为空…...
【鉴权】深入解析 Token:身份认证的核心技术
目录 引言一、Token 的定义与概念1.1 Token 是什么?1.2 无状态性与自包含性1.3 Token 的工作流程1.3.1 基本工作流程1.3.2 工作流程图示 二、Token 的常见用途2.1 用户身份验证2.2 授权控制2.3 防止跨站请求伪造(CSRF)2.4 跨域认证 三、Token…...
FastReport将停止 .NET Framework 上的 WebReport 更新
从2024/ 12 /1 日起,Fastreport将停止发布更新和提供对 FastReport.Web (.NET Framework) 的技术支持。该库一直是使用 Online Designer 的许多项目中报告的核心。这些更改意味着 FastReport.Web (Legacy) 库(FastReport.Net包的一部分)将不再…...
面试:TCP、UDP如何解决丢包问题
文章目录 一、TCP丢包原因、解决办法1.1 TCP为什么会丢包1.2 TCP传输协议如何解决丢包问题1.3 其他丢包情况(拓展)1.4 补充1.4.1 TCP端口号1.4.2 多个TCP请求的逻辑1.4.3 处理大量TCP连接请求的方法1.4.4 总结 二、UDP丢包2.1 UDP协议2.1.1 UDP简介2.1.2…...
在Ubuntu下安装RabbitMQ、添加一个新的登录用户并设置密码
在Ubuntu下安装RabbitMQ、添加一个新的登录用户并设置密码 在Ubuntu下安装RabbitMQ可以按照以下步骤进行:步骤 1: 更新系统步骤 2: 安装Erlang步骤 3: 添加RabbitMQ仓库步骤 4: 更新APT索引并安装RabbitMQ步骤 5: 启动RabbitMQ服务步骤 6: 检查RabbitMQ状态步骤 7: …...
HTTPS通信和TCP通信有什么不一样
HTTPS通信和TCP通信的主要区别如下: 协议层次:HTTPS是应用层协议,建立在HTTP协议之上,并增加了SSL/TLS加密层;而TCP是传输层协议,提供可靠的数据传输服务。安全性:HTTPS通过SSL/TLS加密…...
Kafka 的一些问题,夺命15连问
kafka-中的组成员 kafka四大核心 生产者API 允许应用程序发布记录流至一个或者多个kafka的主题(topics)。 消费者API 允许应用程序订阅一个或者多个主题,并处理这些主题接收到的记录流 StreamsAPI 允许应用程序充当流处理器(s…...
unity3d————延时函数
1.public void InvokeRepeating(string methodName, float time, float repeatRate); 延迟重复执行函数 InvokeRepeating 参数一:函数名字符串 参数二:第一次执行的延迟时间 参数三:之后每次执行的间隔时间 注意: 1-1.延时函数第…...
告别新建工程烦恼:手把手教你为复旦微FM33LE0XX搭建IAR标准库工程模板
复旦微FM33LE0XX开发实战:打造可复用的IAR标准工程模板 在嵌入式开发领域,每次新建项目都从零开始配置工程文件无异于重复造轮子。对于复旦微FM33LE0XX系列单片机开发者而言,一个精心设计的标准工程模板能节省至少80%的初始化时间。本文将带你…...
DanKoe 视频笔记:赚钱是精神性的:破除“高尚的失败者”的迷思 [特殊字符]
在本节课中,我们将要学习一种关于金钱、商业和道德的全新视角。我们将探讨为何将赚钱视为不道德是一种幻觉,以及这种观念如何阻碍你为世界创造真正的价值。 在当今世界,最糟糕的事情之一就是陷入“高尚的失败者”的思维模式。许多聪明人讨厌…...
Web开发全栈实践:搭建展示MiniCPM-V-2_6能力的交互式网站
Web开发全栈实践:搭建展示MiniCPM-V-2_6能力的交互式网站 最近在探索多模态大模型的应用,发现MiniCPM-V-2_6在视觉理解方面表现挺有意思。光看技术文档和跑跑Demo总觉得不过瘾,不如自己动手,用最熟悉的Web技术栈,给它…...
从零搭建Binance Trade Bot:精通加密货币自动交易工具配置与使用
从零搭建Binance Trade Bot:精通加密货币自动交易工具配置与使用 【免费下载链接】binance-trade-bot Automated cryptocurrency trading bot 项目地址: https://gitcode.com/gh_mirrors/bi/binance-trade-bot 一、核心功能解析:Binance Trade Bo…...
Nomic-Embed-Text-V2-MoE部署排错指南:解决403 Forbidden等常见API访问错误
Nomic-Embed-Text-V2-MoE部署排错指南:解决403 Forbidden等常见API访问错误 部署一个新的模型服务,就像给家里添置一台新电器,插上电、打开开关,本以为就能顺利运转,结果却发现指示灯不亮,或者干脆跳闸了。…...
构建智能投资决策中枢:TradingAgents-CN多维度金融分析框架实战指南
构建智能投资决策中枢:TradingAgents-CN多维度金融分析框架实战指南 【免费下载链接】TradingAgents-CN 基于多智能体LLM的中文金融交易框架 - TradingAgents中文增强版 项目地址: https://gitcode.com/GitHub_Trending/tr/TradingAgents-CN 破解投资决策困境…...
Windows右键菜单管理终极指南:3分钟打造高效桌面操作环境
Windows右键菜单管理终极指南:3分钟打造高效桌面操作环境 【免费下载链接】ContextMenuManager 🖱️ 纯粹的Windows右键菜单管理程序 项目地址: https://gitcode.com/gh_mirrors/co/ContextMenuManager 你是否曾因Windows右键菜单过于臃肿而烦恼&…...
Qt图形视图框架性能调优指南:从QGraphicsScene的ItemIndexMethod到视图更新策略
Qt图形视图框架性能调优实战:从索引算法到渲染优化 在开发CAD设计工具或地理信息系统时,我们团队曾遇到一个棘手问题:当场景中的图元数量超过5000个时,平移视图会出现明显的卡顿。通过系统性的性能分析,我们发现90%的C…...
3步解锁VR视频自由:让3D内容在任何设备重生
3步解锁VR视频自由:让3D内容在任何设备重生 【免费下载链接】VR-reversal VR-Reversal - Player for conversion of 3D video to 2D with optional saving of head tracking data and rendering out of 2D copies. 项目地址: https://gitcode.com/gh_mirrors/vr/V…...
Bidili Generator应用场景:电商主图/社交配图/Logo设计一站式生成方案
Bidili Generator应用场景:电商主图/社交配图/Logo设计一站式生成方案 你是不是也遇到过这样的烦恼?做电商,每天要上新几十款商品,每款都得找人设计主图,成本高、周期长;运营社交媒体,天天为找…...
