Go time包AddDate使用解惑实例详解

Go time包AddDate使用解惑实例详解

目录

引例

Go Time 包中是这么处理的

源码分析

预期偏差

怎么解决

结语

我们经常会使用 Go time 包 AddDate(),对日期进行计算。而它得到的结果,可能会往往超出我们的“预期”。(为什么预期要打引号,因为我们的预期可能是模糊、偏差的)。

引例

假设,今天是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

结果输出:20221201,而非我们预期的下个月最后一天11月30日。

Go Time 包中是这么处理的

AddDate() 对月份+1,即变成了11-31,换算成对应的天数、最终换算成对应的纳秒数存储在 Time 对象中;

输出时,Format()将输出标准的日期,Time 中的纳秒会转为 12-01,而不是 11-31,因为这天并不存在;

只要是涉及到大小月的最后一天都会出现这个问题。

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,然后调用Date()处理。

// time/time.go func (t Time) AddDate(years int, months int, days int) Time { year, month, day := t.Date() // 获取当前年月日 hour, min, sec := t.Clock() // 获取当前时分秒 return Date(year+years, month+Month(months), day+days, hour, min, sec, int(t.nsec()), t.Location()) }

Date()中此时传入的参数是

year 2020

month 11

day 31

hour、min、sec、nsec 为运行时的时分秒纳秒

d 计算的是绝对纪元到今天之前的天数:

**d = 今年之前的天数 + 年初到当月之前的天数 + 月初到当天之前的天数;**

最终,将 d 转换成纳秒 + 当天经过的纳秒存储在 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(天数)。两者底层存储的时候都是一样的数据,Format() 时将2022-11-31的Time 格式化成 2022-12-01也就不例外了,输出当然要显示让人看得懂的常规标准日期嘛。

// 2022-11-31 d = 2022年之前的天数 + 1月到10月的总天数 + 30天 // 2022-12-01 d = 2022年之前的天数 + 1月到11月的总天数 + 0天 = 2022年之前的天数 + 1月到10月的总天数 + 30天 + 0天

你甚至可以往 Date() 输入非标准日期2022-11-35,它和标准日期 2022-12-05,将得到同样的 d (天数)。
“非标准日期”和“标准日期”就像天平的两边,虽然形式不一样,但他们实际的质量(d 天数)是一样的。记住这句话,后面有用。

预期偏差

我们弄清楚了原理,但仍然不能接受这个结果。这样的结果是 Go 的 bug 吗?还是 Go Time 包偷懒了?

然而并不是,恰恰是我们的“预期”出现了问题。

正常来说,我们预期 10-30 + 1 month是 11-30 日,这很合理。那我们为什么还期待 10-31 + 1 month 也是 11-30 日?仅仅因为 10-31是当前月的最后一天,我们也期待 +1 month 后是下个月的最后一天吗?

10-30 和 10-31 两个日期相差一天,进行同样的 +1 month 操作后,就变成为了同一天。这就像 1 + 10 = 2 + 10 一样的结果,这显然不合理。

Go 目前的处理结果是正确的,并且他在 AddDate() 注释中也注明了会处理“溢出”的情况。况且,不止 Go 语言这么处理,PHP 也是这么处理的,见文章令人困惑的strtotime

怎么解决

道理我都懂,但我就是想获取上/下一个月的最后一天怎么办?

利用前面源码分析阶段,提到的“天平原理”,就能拿到我们想要的结果。

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 结语

最初,发现这个问题是看鸟哥文章,当时认为那是 PHP 的“坑”,并没有深入思考过。如今,在 Go 语言再次遇到这个问题,重新思考,发现日期函数本应该就那么设计,是我们对日期函数理解不够,产生了错误的“预期”。

以上就是Go time包AddDate使用解惑实例详解的详细内容,更多关于Go time包AddDate的资料请关注易知道(ezd.cc)其它相关文章!

推荐阅读

    计算机主板BIOS设置详细-BIOS知识

    计算机主板BIOS设置详细-BIOS知识,,什么是电脑BIOS,一般电脑主板已经设置完毕后,电脑就开始按del键进入BIOS。系统启动BIOS,即微机的基本输入

    计算机蓝屏故障的计算机蓝屏解决方案

    计算机蓝屏故障的计算机蓝屏解决方案,,电脑蓝屏电脑故障经常使用电脑的朋友经常遇到,因为电脑蓝屏是一个非常普遍的现象,所以很难预测,什么时

    计算机自动关机的原因是什么

    计算机自动关机的原因是什么,,计算机(计算机),通常称为计算机,是一种用于高速计算的电子计算机。它可以进行数值计算和逻辑计算,还具有存储记忆

    电脑功率计算|电脑功率计算公式

    电脑功率计算|电脑功率计算公式,,电脑功率计算公式  从设计角度出发一般取300w/台基本都可以满足要求,可以从以下几个方面分析一下电脑功

    如何设置计算机视图视图的统一视图

    如何设置计算机视图视图的统一视图,,不知道你是否有这样的使用电脑经验,电脑在不同的文件夹打开,有时这个文件夹是用来查看列表的方式,但是当

    Win7电脑屏幕模糊的原因及解决办法

    Win7电脑屏幕模糊的原因及解决办法,解决办法,电脑屏幕,  在办公室工作时,我们总是需要长时间对着电脑,这时候电脑屏幕的清晰度对我们的眼

    的故障_计算机解决无法打印文档

    的故障_计算机解决无法打印文档,,核心提示:最近,打印机出现了一个奇怪的现象,在打印正常之前,打印机不能打印最近的突然,提示发送打印作业,计算

    PC计算机:AMDCPU核心细节

    PC计算机:AMDCPU核心细节,,核心提示:AthlonXP的核心型athlonxp有4种不同的核心类型,但都有个共同点:他们都使用socketa接口,他们都使用PR标称值

    分析计算机减速的原因

    分析计算机减速的原因,,核心提示:做以上九点,我相信你的爱是快的。当然,如果速度很慢,你应该考虑硬件升级。学习电脑组装,就来吧… 有很多人说