欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java實(shí)現(xiàn)優(yōu)雅日期處理的方案詳解

 更新時(shí)間:2025年04月22日 11:32:07   作者:蘇三說(shuō)技術(shù)  
在我們的日常工作中,需要經(jīng)常處理各種格式,各種類(lèi)似的的日期或者時(shí)間,下面我們就來(lái)看看如何使用java處理這樣的日期問(wèn)題吧,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

前言

在我們的日常工作中,需要經(jīng)常處理各種格式,各種類(lèi)似的的日期或者時(shí)間。

比如:2025-04-21、2025/04/21、2025年04月21日等等。

有些字段是String類(lèi)型,有些是Date類(lèi)型,有些是Long類(lèi)型。

如果不同的數(shù)據(jù)類(lèi)型,經(jīng)常需要相互轉(zhuǎn)換,如果處理不好,可能會(huì)出現(xiàn)很多意想不到的問(wèn)題。

這篇文章跟大家一起聊聊日期處理的常見(jiàn)問(wèn)題,和相關(guān)的解決方案,希望對(duì)你會(huì)有所幫助。

一、日期的坑

1.1 日期格式化陷阱

在文章的開(kāi)頭,先給大家列舉一個(gè)非常經(jīng)典的日期格式化問(wèn)題:

// 舊代碼片段(線程不安全的經(jīng)典寫(xiě)法)
public class OrderService {

  private SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");*

    public void saveOrder(Order order) {
        // 線程A和線程B同時(shí)進(jìn)入該方法
        String createTime = sdf.format(order.getCreateTime()); 
        // 可能出現(xiàn)"2023-02-30 12:00:00"這種根本不存在的日期
        orderDao.insert(createTime);**
    }

}

問(wèn)題復(fù)現(xiàn)場(chǎng)景:

  • 高并發(fā)秒殺場(chǎng)景下,10個(gè)線程同時(shí)處理訂單。
  • 每個(gè)線程獲取到的order.getCreateTime()均為2023-02-28 23:59:59。
  • 由于線程調(diào)度順序問(wèn)題,某個(gè)線程執(zhí)行sdf.format()時(shí)。
  • 內(nèi)部Calendar實(shí)例已被其他線程修改為非法狀態(tài)。
  • 最終數(shù)據(jù)庫(kù)中出現(xiàn)2023-02-30這類(lèi)無(wú)效日期。

問(wèn)題根源:SimpleDateFormat內(nèi)部使用了共享的Calendar實(shí)例,多線程并發(fā)修改會(huì)導(dǎo)致數(shù)據(jù)污染。

1.2 時(shí)區(qū)轉(zhuǎn)換

我們?cè)谔幚砣掌诘臅r(shí)候,還可能會(huì)遇到夏令時(shí)轉(zhuǎn)換的問(wèn)題:

// 錯(cuò)誤示范:簡(jiǎn)單加減8小時(shí)
public Date convertToBeijingTime(Date utcDate) {
    Calendar cal = Calendar.getInstance();
    cal.setTime(utcDate);
    cal.add(Calendar.HOUR, 8); // 沒(méi)考慮夏令時(shí)切換問(wèn)題
    return cal.getTime();
}

夏令時(shí)是一種在夏季期間將時(shí)間提前一小時(shí)的制度,旨在充分利用日光,病節(jié)約能源。

在一些國(guó)家和地區(qū),夏令時(shí)的開(kāi)始和結(jié)束時(shí)間是固定的。

而在一些國(guó)家和地區(qū),可能會(huì)根據(jù)需要調(diào)整。

在編程中,我們經(jīng)常需要處理夏令時(shí)轉(zhuǎn)換的問(wèn)題,以確保時(shí)間的正確性。

隱患分析:2024年10月27日北京時(shí)間凌晨2點(diǎn)會(huì)突然跳回1點(diǎn),直接導(dǎo)致訂單時(shí)間計(jì)算錯(cuò)誤

二、優(yōu)雅方案的進(jìn)階之路

2.1 線程安全重構(gòu)

在Java8之前,一般是通過(guò)ThreadLocal解決多線程場(chǎng)景下,日期轉(zhuǎn)換的問(wèn)題。

例如下面這樣:

// ThreadLocal封裝方案(適用于JDK7及以下)
public class SafeDateFormatter {
    private static final ThreadLocal<DateFormat> THREAD_LOCAL = ThreadLocal.withInitial(() -> 
        new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")
    );

    public static String format(Date date) {
        return THREAD_LOCAL.get().format(date);
    }
}

線程安全原理:

  • 每個(gè)線程第一次調(diào)用format()方法時(shí)
  • 會(huì)通過(guò)withInitial()初始化方法創(chuàng)建獨(dú)立的DateFormat實(shí)例
  • 后續(xù)該線程再次調(diào)用時(shí)直接復(fù)用已有實(shí)例
  • 線程銷(xiāo)毀時(shí)會(huì)自動(dòng)清理ThreadLocal存儲(chǔ)的實(shí)例

原理揭秘:通過(guò)ThreadLocal為每個(gè)線程分配獨(dú)立DateFormat實(shí)例,徹底規(guī)避線程安全問(wèn)題。

2.2 Java8時(shí)間API革命

在Java8之后,提供了LocalDateTime類(lèi)對(duì)時(shí)間做轉(zhuǎn)換,它是官方推薦的方案。

例如下面這樣:

// 新時(shí)代寫(xiě)法(線程安全+表達(dá)式增強(qiáng))
public class ModernDateUtils {
    public static String format(LocalDateTime dateTime) {
        return dateTime.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
    }

    public static LocalDateTime parse(String str) {
        return LocalDateTime.parse(str, DateTimeFormatter.ISO_LOCAL_DATE_TIME);
    }
}

黑科技特性

  • 288種預(yù)定義格式器
  • 支持ISO-8601/ZonedDateTime等國(guó)際化標(biāo)準(zhǔn)
  • 不可變對(duì)象天然線程安全

三、高階場(chǎng)景解決方案

3.1 跨時(shí)區(qū)計(jì)算(跨國(guó)公司必備)

下面這個(gè)例子是基于時(shí)區(qū)計(jì)算營(yíng)業(yè)時(shí)長(zhǎng):

// 正確示范:基于時(shí)區(qū)計(jì)算營(yíng)業(yè)時(shí)長(zhǎng)
public Duration calculateBusinessHours(ZonedDateTime start, ZonedDateTime end) {
    // 顯式指定時(shí)區(qū)避免歧義
    ZonedDateTime shanghaiStart = start.withZoneSameInstant(ZoneId.of("Asia/Shanghai"));
    ZonedDateTime newYorkEnd = end.withZoneSameInstant(ZoneId.of("America/New_York"));
    
    // 自動(dòng)處理夏令時(shí)切換
    return Duration.between(shanghaiStart, newYorkEnd);
}

底層原理:通過(guò)ZoneId維護(hù)完整的時(shí)區(qū)規(guī)則庫(kù)(含歷史變更數(shù)據(jù)),自動(dòng)處理夏令時(shí)切換。

3.2 性能優(yōu)化實(shí)戰(zhàn)

日均億級(jí)請(qǐng)求的處理方案:

// 預(yù)編譯模式(性能提升300%)
public class CachedDateFormatter {
    private static final Map<String, DateTimeFormatter> CACHE = new ConcurrentHashMap<>();

    public static DateTimeFormatter getFormatter(String pattern) {
        return CACHE.computeIfAbsent(pattern, DateTimeFormatter::ofPattern);
    }
}

我們可以使用static final這種預(yù)編譯模式,來(lái)提升日期轉(zhuǎn)換的性能。

性能對(duì)比

方案內(nèi)存占用初始化耗時(shí)格式化速度
每次新建Formatter1.2GB2.3s1200 req/s
預(yù)編譯緩存230MB0.8s5800 req/s

3.3 全局時(shí)區(qū)上下文+攔截器

為了方便統(tǒng)一解決時(shí)區(qū)問(wèn)題,我們可以使用全局時(shí)區(qū)上下文+攔截器。

例如下面這樣:

// 全局時(shí)區(qū)上下文傳遞
public class TimeZoneContext {
    private static final ThreadLocal<ZoneId> CONTEXT_HOLDER = new ThreadLocal<>();

    public static void setTimeZone(ZoneId zoneId) {
        CONTEXT_HOLDER.set(zoneId);
    }

    public static ZoneId getTimeZone() {
        return CONTEXT_HOLDER.get();
    }
}

// 在Spring Boot攔截器中設(shè)置時(shí)區(qū)
@Component
public class TimeZoneInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        String timeZoneId = request.getHeader("X-Time-Zone");
        TimeZoneContext.setTimeZone(ZoneId.of(timeZoneId));
        return true;
    }
}

此外,還需要在請(qǐng)求接口的header中傳遞X-Time-Zone時(shí)區(qū)參數(shù)。

四、優(yōu)雅設(shè)計(jì)的底層邏輯

4.1 不可變性原則

// LocalDate的不可變?cè)O(shè)計(jì)
LocalDate date = LocalDate.now();
date.plusDays(1); // 返回新實(shí)例,原對(duì)象不變
System.out.println(date); // 輸出當(dāng)前日期,不受影響

4.2 函數(shù)式編程思維

// Stream API處理時(shí)間序列
List<Transaction> transactions = 
    list.stream()
        .filter(t -> t.getTimestamp().isAfter(yesterday)) // 聲明式過(guò)濾
        .sorted(Comparator.comparing(Transaction::getTimestamp)) // 自然排序
        .collect(Collectors.toList()); // 延遲執(zhí)行

五、總結(jié)

下面總結(jié)一下日期處理的各種方案:

境界代碼特征典型問(wèn)題修復(fù)成本
初級(jí)大量使用String拼接格式混亂/解析異常
進(jìn)階熟練運(yùn)用JDK8新API時(shí)區(qū)處理不當(dāng)
高手預(yù)編譯+緩存+防御性編程性能瓶頸
大師結(jié)合領(lǐng)域模型設(shè)計(jì)時(shí)間類(lèi)型業(yè)務(wù)邏輯漏洞極低

終極建議:在微服務(wù)架構(gòu)中,建議建立統(tǒng)一的時(shí)間處理中間件,通過(guò)AOP攔截所有時(shí)間相關(guān)操作,徹底消除代碼層面的時(shí)間處理差異。

到此這篇關(guān)于Java實(shí)現(xiàn)優(yōu)雅日期處理的方案詳解的文章就介紹到這了,更多相關(guān)Java日期處理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java實(shí)現(xiàn)驗(yàn)證碼具體代碼

    Java實(shí)現(xiàn)驗(yàn)證碼具體代碼

    這篇文章主要介紹了Java實(shí)現(xiàn)驗(yàn)證碼具體代碼,有需要的朋友可以參考一下
    2013-12-12
  • Java中動(dòng)態(tài)規(guī)則的實(shí)現(xiàn)方式示例詳解

    Java中動(dòng)態(tài)規(guī)則的實(shí)現(xiàn)方式示例詳解

    這篇文章主要介紹了Java中動(dòng)態(tài)規(guī)則的實(shí)現(xiàn)方式,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-08-08
  • Java對(duì)象的內(nèi)存布局詳細(xì)介紹

    Java對(duì)象的內(nèi)存布局詳細(xì)介紹

    這篇文章主要介紹了Java對(duì)象的內(nèi)存布局,我們知道在Java中基本數(shù)據(jù)類(lèi)型的大小,例如int類(lèi)型占4個(gè)字節(jié)、long類(lèi)型占8個(gè)字節(jié),那么Integer對(duì)象和Long對(duì)象會(huì)占用多少內(nèi)存呢?本文介紹一下Java對(duì)象在堆中的內(nèi)存結(jié)構(gòu)以及對(duì)象大小的計(jì)算
    2023-02-02
  • Java線程池的拒絕策略實(shí)現(xiàn)詳解

    Java線程池的拒絕策略實(shí)現(xiàn)詳解

    這篇文章主要介紹了Java線程池的拒絕策略實(shí)現(xiàn)詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-09-09
  • 解決安裝mysqlclient的時(shí)候出現(xiàn)Microsoft Visual C++ 14.0 is required報(bào)錯(cuò)

    解決安裝mysqlclient的時(shí)候出現(xiàn)Microsoft Visual C++ 14.0 is required報(bào)錯(cuò)

    這篇文章主要介紹了解決安裝mysqlclient的時(shí)候出現(xiàn)Microsoft Visual C++ 14.0 is required報(bào)錯(cuò)問(wèn)題,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2018-11-11
  • Spring多線程的使用以及問(wèn)題詳解

    Spring多線程的使用以及問(wèn)題詳解

    在我們開(kāi)發(fā)系統(tǒng)過(guò)程中,經(jīng)常會(huì)處理一些費(fèi)時(shí)間的任務(wù)(如:向數(shù)據(jù)庫(kù)中插入大量數(shù)據(jù)),這個(gè)時(shí)候就就需要使用多線程,下面這篇文章主要給大家介紹了關(guān)于Spring多線程的使用以及問(wèn)題的相關(guān)資料,需要的朋友可以參考下
    2022-05-05
  • Hibernate Validator實(shí)現(xiàn)更簡(jiǎn)潔的參數(shù)校驗(yàn)及一個(gè)util

    Hibernate Validator實(shí)現(xiàn)更簡(jiǎn)潔的參數(shù)校驗(yàn)及一個(gè)util

    這篇文章主要介紹了Hibernate Validator實(shí)現(xiàn)更簡(jiǎn)潔的參數(shù)校驗(yàn)及一個(gè)util,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2019-05-05
  • 使用Java生成32位16進(jìn)制密鑰的代碼實(shí)現(xiàn)

    使用Java生成32位16進(jìn)制密鑰的代碼實(shí)現(xiàn)

    在許多加密和安全應(yīng)用中,生成隨機(jī)的密鑰是至關(guān)重要的一步,密鑰通常以16進(jìn)制形式表示,并且具有特定的長(zhǎng)度,在這篇博客中,我們將探討如何使用Java生成一個(gè)32位長(zhǎng)度的16進(jìn)制密鑰,并展示詳細(xì)的代碼示例和運(yùn)行結(jié)果,需要的朋友可以參考下
    2024-08-08
  • 詳解Java的Spring框架中bean的注入集合

    詳解Java的Spring框架中bean的注入集合

    這篇文章主要介紹了詳解Java的Spring框架中bean的注入集合,Spring是Java的SSH三大web開(kāi)發(fā)框架之一,需要的朋友可以參考下
    2015-12-12
  • SpringBoot定義Bean的幾種實(shí)現(xiàn)方式

    SpringBoot定義Bean的幾種實(shí)現(xiàn)方式

    本文主要介紹了SpringBoot定義Bean的幾種實(shí)現(xiàn)方式,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05

最新評(píng)論