成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

瞧瞧別人家的日期處理,那叫一個優雅!

開發 前端
如果不同的數據類型,經常需要相互轉換,如果處理不好,可能會出現很多意想不到的問題。這篇文章跟大家一起聊聊日期處理的常見問題,和相關的解決方案,希望對你會有所幫助。

前言

在我們的日常工作中,需要經常處理各種格式,各種類似的的日期或者時間。

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

有些字段是String類型,有些是Date類型,有些是Long類型。

如果不同的數據類型,經常需要相互轉換,如果處理不好,可能會出現很多意想不到的問題。

這篇文章跟大家一起聊聊日期處理的常見問題,和相關的解決方案,希望對你會有所幫助。

一、日期的坑

1.1 日期格式化陷阱

在文章的開頭,先給大家列舉一個非常經典的日期格式化問題:

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

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

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

}

問題復現場景:

  • 高并發秒殺場景下,10個線程同時處理訂單。
  • 每個線程獲取到的order.getCreateTime()均為2023-02-28 23:59:59。
  • 由于線程調度順序問題,某個線程執行sdf.format()時。
  • 內部Calendar實例已被其他線程修改為非法狀態。
  • 最終數據庫中出現2023-02-30這類無效日期。

問題根源:SimpleDateFormat內部使用了共享的Calendar實例,多線程并發修改會導致數據污染。

1.2 時區轉換

我們在處理日期的時候,還可能會遇到夏令時轉換的問題:

// 錯誤示范:簡單加減8小時
public Date convertToBeijingTime(Date utcDate) {
    Calendar cal = Calendar.getInstance();
    cal.setTime(utcDate);
    cal.add(Calendar.HOUR, 8); // 沒考慮夏令時切換問題
    return cal.getTime();
}

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

在一些國家和地區,夏令時的開始和結束時間是固定的。

而在一些國家和地區,可能會根據需要調整。

在編程中,我們經常需要處理夏令時轉換的問題,以確保時間的正確性。

隱患分析:2024年10月27日北京時間凌晨2點會突然跳回1點,直接導致訂單時間計算錯誤

二、優雅方案的進階之路

2.1 線程安全重構

在Java8之前,一般是通過ThreadLocal解決多線程場景下,日期轉換的問題。

例如下面這樣:

// 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);
    }
}

線程安全原理:

  1. 每個線程第一次調用format()方法時
  2. 會通過withInitial()初始化方法創建獨立的DateFormat實例
  3. 后續該線程再次調用時直接復用已有實例
  4. 線程銷毀時會自動清理ThreadLocal存儲的實例

原理揭秘:通過ThreadLocal為每個線程分配獨立DateFormat實例,徹底規避線程安全問題。

2.2 Java8時間API革命

在Java8之后,提供了LocalDateTime類對時間做轉換,它是官方推薦的方案。

例如下面這樣:

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

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

黑科技特性:

  • 288種預定義格式器
  • 支持ISO-8601/ZonedDateTime等國際化標準
  • 不可變對象天然線程安全

三、高階場景解決方案

3.1 跨時區計算(跨國公司必備)

下面這個例子是基于時區計算營業時長:

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

底層原理:通過ZoneId維護完整的時區規則庫(含歷史變更數據),自動處理夏令時切換。

3.2 性能優化實戰

日均億級請求的處理方案:

// 預編譯模式(性能提升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這種預編譯模式,來提升日期轉換的性能。

性能對比:

方案

內存占用

初始化耗時

格式化速度

每次新建Formatter

1.2GB

2.3s

1200 req/s

預編譯緩存

230MB

0.8s

5800 req/s

3.3 全局時區上下文+攔截器

為了方便統一解決時區問題,我們可以使用全局時區上下文+攔截器。

例如下面這樣:

// 全局時區上下文傳遞
publicclass TimeZoneContext {
    privatestaticfinal 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攔截器中設置時區
@Component
publicclass 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));
        returntrue;
    }
}

此外,還需要在請求接口的header中傳遞X-Time-Zone時區參數。

四、優雅設計的底層邏輯

4.1 不可變性原則

// LocalDate的不可變設計
LocalDate date = LocalDate.now();
date.plusDays(1); // 返回新實例,原對象不變
System.out.println(date); // 輸出當前日期,不受影響

4.2 函數式編程思維

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

五、總結

下面總結一下日期處理的各種方案:

境界

代碼特征

典型問題

修復成本

初級

大量使用String拼接

格式混亂/解析異常

進階

熟練運用JDK8新API

時區處理不當

高手

預編譯+緩存+防御性編程

性能瓶頸

大師

結合領域模型設計時間類型

業務邏輯漏洞

極低

終極建議:在微服務架構中,建議建立統一的時間處理中間件,通過AOP攔截所有時間相關操作,徹底消除代碼層面的時間處理差異。

最后跟大家分享一下,日期處理的架構演進路線圖:

圖片圖片

責任編輯:武曉燕 來源: 蘇三說技術
相關推薦

2024-10-24 08:21:33

2024-11-12 08:20:31

2025-05-30 08:20:54

2025-04-08 08:20:33

2022-12-12 08:14:47

2024-12-02 00:59:30

Spring

2025-03-06 08:21:02

判空entity對象

2025-02-28 08:21:00

2025-03-11 08:20:58

2020-11-03 16:00:33

API接口微服務框架編程語言

2015-09-24 09:22:16

nodejs頁面始末

2017-11-12 21:32:52

戴爾

2016-01-08 09:49:19

DockerDocker案例云應用開發

2020-11-17 09:34:31

API接口后端

2017-09-22 13:22:59

大數據南京大學宿舍

2021-07-14 06:31:08

京東互聯網加薪

2023-12-30 20:04:51

MyBatis框架數據

2021-01-20 05:42:27

RabbitMQMQ vhost

2017-06-13 14:15:51

戴爾協同計算汽車神話

2019-06-11 09:35:50

戴爾
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲精品国产第一综合99久久 | 999视频| 久久精品青青大伊人av | 欧洲一级黄| 新超碰97 | 在线观看亚洲精品视频 | 91看片官网 | 精品国产一区二区三区久久久久久 | 天天操夜夜操 | 毛片1| 黄色一级大片在线免费看产 | 亚洲精品68久久久一区 | 一级毛片黄片 | 男人av的天堂 | 人人玩人人添人人澡欧美 | 男女在线网站 | 男女羞羞视频在线免费观看 | 成人不卡 | 99久久99 | 国产精品福利网站 | 午夜精品一区二区三区三上悠亚 | 亚洲精品日本 | 中文字幕日韩欧美一区二区三区 | 亚洲一区二区在线视频 | 毛片免费在线 | 国产亚洲高清视频 | 久久国产一区 | 999热视频 | 国产精品亚洲综合 | 国产高清在线观看 | 亚洲国产一区视频 | 最近中文字幕免费 | 午夜在线免费观看视频 | 成人精品国产免费网站 | 午夜视频在线观看网站 | 欧美成人综合 | 精品成人免费一区二区在线播放 | 久久成人一区 | 午夜资源| 国产乱码精品一区二区三区忘忧草 | 国产亚洲精品久久久久久牛牛 |