網(wǎng)站首頁 編程語言 正文
我們經(jīng)常會使用 Go time 包?AddDate()
,對日期進行計算。而它得到的結(jié)果,可能會往往超出我們的“預(yù)期”。(為什么預(yù)期要打引號,因為我們的預(yù)期可能是模糊、偏差的)。
引例
假設(shè),今天是10月31日,是10月的最后一天,我們想通過?AddDate()
計算下個月的最后一天。
today := time.Date(2022, 10, 31, 0, 0, 0, 0, time.Local) nextDay := today.AddDate(0, 1, 0) fmt.Println(nextDay.Format("20060102")) // 輸出:20221201
結(jié)果輸出:20221201
,而非我們預(yù)期的下個月最后一天11月30日。
Go Time 包中是這么處理的
-
AddDate()
?對月份+1,即變成了11-31,換算成對應(yīng)的天數(shù)、最終換算成對應(yīng)的納秒數(shù)存儲在 Time 對象中; - 輸出時,
Format()
將輸出標(biāo)準(zhǔn)的日期,Time 中的納秒會轉(zhuǎn)為 12-01,而不是 11-31,因為這天并不存在;
只要是涉及到大小月的最后一天都會出現(xiàn)這個問題。
today := time.Date(2022, 3, 31, 0, 0, 0, 0, time.Local) d := today.AddDate(0, -1, 0) fmt.Println(d.Format("20060102")) // 20220303 today := time.Date(2022, 3, 31, 0, 0, 0, 0, time.Local) d := today.AddDate(0, 1, 0) fmt.Println(d.Format("20060102")) // 20220501 today := time.Date(2022, 10, 31, 0, 0, 0, 0, time.Local) d := today.AddDate(0, -1, 0) fmt.Println(d.Format("20060102")) // 20221001 today := time.Date(2022, 10, 31, 0, 0, 0, 0, time.Local) d := today.AddDate(0, 1, 0) fmt.Println(d.Format("20060102")) // 20221201
源碼分析
看一下 Go Time 包具體源碼,仍以開頭10-31 + 1 month
的例子為用例。AddDate()
,首先對?month+1
,然后調(diào)用Date()
處理。
// time/time.go func (t Time) AddDate(years int, months int, days int) Time { year, month, day := t.Date() // 獲取當(dāng)前年月日 hour, min, sec := t.Clock() // 獲取當(dāng)前時分秒 return Date(year+years, month+Month(months), day+days, hour, min, sec, int(t.nsec()), t.Location()) }
Date()
中此時傳入的參數(shù)是
- year 2020
- month 11
- day 31
- hour、min、sec、nsec 為運行時的時分秒納秒
d 計算的是絕對紀(jì)元到今天之前的天數(shù):
**d = 今年之前的天數(shù) + 年初到當(dāng)月之前的天數(shù) + 月初到當(dāng)天之前的天數(shù);**
最終,將?d 轉(zhuǎn)換成納秒?+?當(dāng)天經(jīng)過的納秒存儲在 Time 對象中。
// time/time.go func Date(year int, month Month, day, hour, min, sec, nsec int, loc *Location) Time { …… // Compute days since the absolute epoch. d := daysSinceEpoch(year) // Add in days before this month. d += uint64(daysBefore[month-1]) if isLeap(year) && month >= March { d++ // February 29 } // Add in days before today. d += uint64(day - 1) // Add in time elapsed today. abs := d * secondsPerDay abs += uint64(hour*secondsPerHour + min*secondsPerMinute + sec) …… return t }
對 Date() 輸入2022-11-31
和輸入2022-12-01
,將得到同樣的 d(天數(shù))。兩者底層存儲的時候都是一樣的數(shù)據(jù),F(xiàn)ormat() 時將2022-11-31
的Time 格式化成?2022-12-01
也就不例外了,輸出當(dāng)然要顯示讓人看得懂的常規(guī)標(biāo)準(zhǔn)日期嘛。
// 2022-11-31 d = 2022年之前的天數(shù) + 1月到10月的總天數(shù) + 30天 // 2022-12-01 d = 2022年之前的天數(shù) + 1月到11月的總天數(shù) + 0天 = 2022年之前的天數(shù) + 1月到10月的總天數(shù) + 30天 + 0天
你甚至可以往 Date() 輸入非標(biāo)準(zhǔn)日期2022-11-35
,它和標(biāo)準(zhǔn)日期?2022-12-05
,將得到同樣的 d (天數(shù))。
“非標(biāo)準(zhǔn)日期”和“標(biāo)準(zhǔn)日期”就像天平的兩邊,雖然形式不一樣,但他們實際的質(zhì)量(d 天數(shù))是一樣的。記住這句話,后面有用。
預(yù)期偏差
我們弄清楚了原理,但仍然不能接受這個結(jié)果。這樣的結(jié)果是 Go 的 bug 嗎?還是 Go Time 包偷懶了?
然而并不是,恰恰是我們的“預(yù)期”出現(xiàn)了問題。
正常來說,我們預(yù)期?10-30
?+?1 month
是?11-30
?日,這很合理。那我們?yōu)槭裁催€期待?10-31
?+?1 month
?也是?11-30
?日?僅僅因為?10-31
是當(dāng)前月的最后一天,我們也期待?+1 month
?后是下個月的最后一天嗎?
10-30 和 10-31 兩個日期相差一天,進行同樣的?+1 month
?操作后,就變成為了同一天。這就像 1 + 10 = 2 + 10 一樣的結(jié)果,這顯然不合理。
Go 目前的處理結(jié)果是正確的,并且他在?AddDate()
?注釋中也注明了會處理“溢出”的情況。況且,不止 Go 語言這么處理,PHP 也是這么處理的,見文章令人困惑的strtotime
怎么解決
道理我都懂,但我就是想獲取上/下一個月的最后一天怎么辦?
利用前面源碼分析階段,提到的“天平原理”,就能拿到我們想要的結(jié)果。
today := time.Date(2022, 10, 31, 0, 0, 0, 0, time.Local) d := today.Day() // 上個月最后一天 // 10-00 日 等于 9-30 日 day1 := today.AddDate(0, 0, -d) fmt.Println(day1.Format("20060102")) // 下個月最后一天 // 12-00 日 等于 11-30 日 day2 := today.AddDate(0, 2, -d) fmt.Println(day2.Format("20060102")) // 20220930 // 20221130
結(jié)語
最初,發(fā)現(xiàn)這個問題是看鳥哥文章,當(dāng)時認(rèn)為那是 PHP 的“坑”,并沒有深入思考過。如今,在 Go 語言再次遇到這個問題,重新思考,發(fā)現(xiàn)日期函數(shù)本應(yīng)該就那么設(shè)計,是我們對日期函數(shù)理解不夠,產(chǎn)生了錯誤的“預(yù)期”。
原文鏈接:https://segmentfault.com/a/1190000042502580
相關(guān)推薦
- 2022-07-15 使用Android原生WebView+Highcharts實現(xiàn)可左右滑動的折線圖_Android
- 2022-03-24 聊一聊redis奇葩數(shù)據(jù)類型與集群知識_Redis
- 2022-06-12 Python數(shù)據(jù)傳輸黏包問題_python
- 2022-01-06 Spring Aware接口詳解
- 2022-05-23 iOS?UISegmentControl實現(xiàn)自定義分欄效果_IOS
- 2022-04-19 前端開發(fā)中幾種存儲方式詳解
- 2022-08-11 Go語言Grpc?Stream的實現(xiàn)_Golang
- 2022-07-27 C++中整形與浮點型如何在內(nèi)存中的存儲詳解_C 語言
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支