前言:Time.ISOWeek() 返回您从星期一开始的周数,因此我将回答您的问题,该问题也处理从星期一开始的周数。如果您希望它在周日开始的几周内工作,请根据您的需要进行更改。
我在github.com/icza/gox 发布了这个实用程序,请参阅timex.WeekStart()。
标准库不提供返回给定周的日期范围(年+周数)的函数。所以我们必须自己构建一个。
这并不难。我们可以从年中开始,与一周的第一天(星期一)对齐,获取该时间值所在的周,然后修正:添加与周差乘以 7 一样多的天。
这就是它的样子:
func WeekStart(year, week int) time.Time {
// Start from the middle of the year:
t := time.Date(year, 7, 1, 0, 0, 0, 0, time.UTC)
// Roll back to Monday:
if wd := t.Weekday(); wd == time.Sunday {
t = t.AddDate(0, 0, -6)
} else {
t = t.AddDate(0, 0, -int(wd)+1)
}
// Difference in weeks:
_, w := t.ISOWeek()
t = t.AddDate(0, 0, (week-w)*7)
return t
}
测试它:
fmt.Println(WeekStart(2018, 1))
fmt.Println(WeekStart(2018, 2))
fmt.Println(WeekStart(2019, 1))
fmt.Println(WeekStart(2019, 2))
输出(在Go Playground上试试):
2018-01-01 00:00:00 +0000 UTC
2018-01-08 00:00:00 +0000 UTC
2018-12-31 00:00:00 +0000 UTC
2019-01-07 00:00:00 +0000 UTC
这个WeekStart() 实现的一个很好的特性是它可以很好地处理超出范围 周。即如果一周通过0,将被解释为上一年的最后一周。如果一周通过-1,它将指定上一年的倒数第二周。同样,如果您通过一年中的最大周数加 1,它将被解释为下一年的第一周等。
上面的WeekStart()函数只返回给定一周的第一天(星期一),因为一周的最后一天总是它的第一天+6天。
如果我们还需要最后一天:
func WeekRange(year, week int) (start, end time.Time) {
start = WeekStart(year, week)
end = start.AddDate(0, 0, 6)
return
}
测试它:
fmt.Println(WeekRange(2018, 1))
fmt.Println(WeekRange(2018, 2))
fmt.Println(WeekRange(2019, 1))
fmt.Println(WeekRange(2019, 2))
输出(在Go Playground上试试):
2018-01-01 00:00:00 +0000 UTC 2018-01-07 00:00:00 +0000 UTC
2018-01-08 00:00:00 +0000 UTC 2018-01-14 00:00:00 +0000 UTC
2018-12-31 00:00:00 +0000 UTC 2019-01-06 00:00:00 +0000 UTC
2019-01-07 00:00:00 +0000 UTC 2019-01-13 00:00:00 +0000 UTC