溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Go?time包AddDate怎么使用

發(fā)布時(shí)間:2022-09-22 09:42:31 來(lái)源:億速云 閱讀:155 作者:iii 欄目:開(kāi)發(fā)技術(shù)

這篇文章主要講解了“Go time包AddDate怎么使用”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“Go time包AddDate怎么使用”吧!

    我們經(jīng)常會(huì)使用 Go time 包 AddDate(),對(duì)日期進(jìn)行計(jì)算。而它得到的結(jié)果,可能會(huì)往往超出我們的“預(yù)期”。(為什么預(yù)期要打引號(hào),因?yàn)槲覀兊念A(yù)期可能是模糊、偏差的)。

    引例

    假設(shè),今天是10月31日,是10月的最后一天,我們想通過(guò) AddDate()計(jì)算下個(gè)月的最后一天。

    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ù)期的下個(gè)月最后一天11月30日。

    Go Time 包中是這么處理的

    • AddDate() 對(duì)月份+1,即變成了11-31,換算成對(duì)應(yīng)的天數(shù)、最終換算成對(duì)應(yīng)的納秒數(shù)存儲(chǔ)在 Time 對(duì)象中;

    • 輸出時(shí),Format()將輸出標(biāo)準(zhǔn)的日期,Time 中的納秒會(huì)轉(zhuǎn)為 12-01,而不是 11-31,因?yàn)檫@天并不存在;

    只要是涉及到大小月的最后一天都會(huì)出現(xiàn)這個(gè)問(wè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 包具體源碼,仍以開(kāi)頭10-31 + 1 month的例子為用例。
    AddDate(),首先對(duì) 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)前時(shí)分秒
        return Date(year+years, month+Month(months), day+days, hour, min, sec, int(t.nsec()), t.Location())
    }

    Date()中此時(shí)傳入的參數(shù)是

    • year 2020

    • month 11

    • day 31

    • hour、min、sec、nsec 為運(yùn)行時(shí)的時(shí)分秒納秒

    d 計(jì)算的是絕對(duì)紀(jì)元到今天之前的天數(shù):

    **d = 今年之前的天數(shù) + 年初到當(dāng)月之前的天數(shù) + 月初到當(dāng)天之前的天數(shù);**

    最終,將 d 轉(zhuǎn)換成納秒 + 當(dāng)天經(jīng)過(guò)的納秒存儲(chǔ)在 Time 對(duì)象中。

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

    對(duì) Date() 輸入2022-11-31和輸入2022-12-01,將得到同樣的 d(天數(shù))。兩者底層存儲(chǔ)的時(shí)候都是一樣的數(shù)據(jù),F(xiàn)ormat() 時(shí)將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)日期”就像天平的兩邊,雖然形式不一樣,但他們實(shí)際的質(zhì)量(d 天數(shù))是一樣的。記住這句話,后面有用。

    預(yù)期偏差

    我們弄清楚了原理,但仍然不能接受這個(gè)結(jié)果。這樣的結(jié)果是 Go 的 bug 嗎?還是 Go Time 包偷懶了?

    然而并不是,恰恰是我們的“預(yù)期”出現(xiàn)了問(wèn)題。

    正常來(lái)說(shuō),我們預(yù)期 10-30 + 1 month是 11-30 日,這很合理。那我們?yōu)槭裁催€期待 10-31 + 1 month 也是 11-30 日??jī)H僅因?yàn)?nbsp;10-31是當(dāng)前月的最后一天,我們也期待 +1 month 后是下個(gè)月的最后一天嗎?

    10-30 和 10-31 兩個(gè)日期相差一天,進(jìn)行同樣的 +1 month 操作后,就變成為了同一天。這就像 1 + 10 = 2 + 10 一樣的結(jié)果,這顯然不合理。

    Go 目前的處理結(jié)果是正確的,并且他在 AddDate() 注釋中也注明了會(huì)處理“溢出”的情況。況且,不止 Go 語(yǔ)言這么處理,PHP 也是這么處理的,見(jiàn)文章令人困惑的strtotime

    怎么解決

    道理我都懂,但我就是想獲取上/下一個(gè)月的最后一天怎么辦?

    利用前面源碼分析階段,提到的“天平原理”,就能拿到我們想要的結(jié)果。

    today := time.Date(2022, 10, 31, 0, 0, 0, 0, time.Local)
    d := today.Day()
    // 上個(gè)月最后一天
    // 10-00 日 等于 9-30 日
    day1 := today.AddDate(0, 0, -d)
    fmt.Println(day1.Format("20060102"))
    // 下個(gè)月最后一天
    // 12-00 日 等于 11-30 日
    day2 := today.AddDate(0, 2, -d)
    fmt.Println(day2.Format("20060102"))
    // 20220930
    // 20221130

    感謝各位的閱讀,以上就是“Go time包AddDate怎么使用”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)Go time包AddDate怎么使用這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

    向AI問(wèn)一下細(xì)節(jié)

    免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

    AI