Go基础包系列 | Time包相关
Time包相关
Time
时间和日期是我们编程中经常会用到的,本文主要介绍了Go语言内置的time包的基本用法。
1 time包
time包提供了时间的显示和测量用的函数。日历的计算采用的是公历。
2 时间类型
time.Time类型表示时间。我们可以通过time.Now()函数获取当前的时间对象,然后获取时间对象的年月日时分秒等信息。示例代码如下:
func timeDemo() {
now := time.Now() //获取当前时间
fmt.Printf("current time:%v\n", now)
year := now.Year() //年
month := now.Month() //月
day := now.Day() //日
hour := now.Hour() //小时
minute := now.Minute() //分钟
second := now.Second() //秒
fmt.Printf("%d-%02d-%02d %02d:%02d:%02d\n", year, month, day, hour, minute, second)
}
3 时间戳
时间戳是自1970年1月1日(08:00:00GMT)至当前时间的总毫秒数。它也被称为Unix时间戳(UnixTimestamp)。
基于时间对象获取时间戳的示例代码如下:
func timestampDemo() {
now := time.Now() //获取当前时间
timestamp1 := now.Unix() //时间戳
timestamp2 := now.UnixNano() //纳秒时间戳
fmt.Printf("current timestamp1:%v\n", timestamp1)
fmt.Printf("current timestamp2:%v\n", timestamp2)
}
使用time.Unix()函数可以将时间戳转为时间格式。
func timestampDemo2(timestamp int64) {
timeObj := time.Unix(timestamp, 0) //将时间戳转为时间格式
fmt.Println(timeObj)
year := timeObj.Year() //年
month := timeObj.Month() //月
day := timeObj.Day() //日
hour := timeObj.Hour() //小时
minute := timeObj.Minute() //分钟
second := timeObj.Second() //秒
fmt.Printf("%d-%02d-%02d %02d:%02d:%02d\n", year, month, day, hour, minute, second)
}
4. 时间间隔
time.Duration是time包定义的一个类型,它代表两个时间点之间经过的时间,以纳秒为单位。time.Duration表示一段时间间隔,可表示的最长时间段大约290年。
time包中定义的时间间隔类型的常量如下:
const (
Nanosecond Duration = 1
Microsecond = 1000 * Nanosecond
Millisecond = 1000 * Microsecond
Second = 1000 * Millisecond
Minute = 60 * Second
Hour = 60 * Minute
)
例如:time.Duration表示1纳秒,time.Second表示1秒。
5 时间操作
Add
我们在日常的编码过程中可能会遇到要求时间+时间间隔的需求,Go语言的时间对象有提供Add方法如下:
func (t Time) Add(d Duration) Time
举个例子,求一个小时之后的时间:
func main() {
now := time.Now()
later := now.Add(time.Hour) // 当前时间加1小时后的时间
fmt.Println(later)
}
Sub
求两个时间之间的差值:
func (t Time) Sub(u Time) Duration
返回一个时间段t-u。如果结果超出了Duration可以表示的最大值/最小值,将返回最大值/最小值。要获取时间点t-d(d为Duration),可以使用t.Add(-d)。
Equal
func (t Time) Equal(u Time) bool
判断两个时间是否相同,会考虑时区的影响,因此不同时区标准的时间也可以正确比较。本方法和用t==u不同,这种方法还会比较地点和时区信息。
Before
func (t Time) Before(u Time) bool
如果t代表的时间点在u之前,返回真;否则返回假。
After
func (t Time) After(u Time) bool
如果t代表的时间点在u之后,返回真;否则返回假。
6 定时器
使用time.Tick(时间间隔)来设置定时器,定时器的本质上是一个通道(channel)。
func tickDemo() { // 这个范例适用于不需要关闭,会导致泄露
ticker := time.Tick(time.Second) //定义一个1秒间隔的定时器
for i := range ticker {
fmt.Println(i)//每秒都会执行的任务
}
}
“泄露”的Tick:
// Tick是NewTicker的方便包装器,提供对Tick的访问通道。
// 而Tick对于不需要关闭的客户端是有用的
// 请注意,如果没有办法关闭它的底层,垃圾回收器不能回收代码,它会“泄漏”。
// 与NewTicker不同(panic错误),如果d <= 0, Tick将返回nil。
func Tick(d Duration) <-chan Time {
if d <= 0 {
return nil
}
return NewTicker(d).C
}
定时器注意事项
如何优雅的关闭定时器:
要注意在go里面使用defer关闭而不是在外面使用stop,这样会导致通道不关闭的
错误示范:
缺少结束标志信息!
func UseTickerWrong() *time.Ticker {
ticker := time.NewTicker(5 * time.Second)
go func(ticker *time.Ticker) {
for range ticker.C {
fmt.Println("hello")
}
fmt.Println("Ticker is Stop")
}(ticker)
return ticker
}
// main
func main() {
ticker1 := UseTickerWrong()
time.Sleep(20 * time.Second)
ticker1.Stop() // 外面关闭会导致读通道没关闭,最后失效
}
正确操作:
// 优雅地使用ticker定时器
type MyTicker struct {
stopChan chan<- bool
}
func NewTicker(duration time.Duration, f func()) *MyTicker {
if duration <= 0 { // 时间不符合规范,返回nil
return nil
}
ticker := time.NewTicker(duration)
stopChan := make(chan bool) // 结束标志
go func(ticker *time.Ticker) {
defer ticker.Stop()
for {
select {
case <-ticker.C:
f()
case stop := <-stopChan:
if stop {
fmt.Println("Ticker is Stop")
return
}
}
}
}(ticker)
return &MyTicker{stopChan}
}
func (t *MyTicker) CloseTicker() {
t.stopChan <- true
close(t.stopChan)
}
func main() {
ticker := NewTicker(time.Second, func() {
fmt.Println("hello")
})
time.Sleep(5 * time.Second)
ticker.CloseTicker()
}
7 时间格式化
时间类型有一个自带的方法Format进行格式化,需要注意的是Go语言中格式化时间模板不是常见的Y-m-d H:M:S而是使用Go的诞生时间2006年1月2号15点04分(记忆口诀为2006 1 2 3 4)。
补充:如果想格式化为12小时方式,需指定PM。
func formatDemo() {
now := time.Now()
// 格式化的模板为Go的出生时间2006年1月2号15点04分 Mon Jan
// 24小时制
fmt.Println(now.Format("2006-01-02 15:04:05.000 Mon Jan"))
// 12小时制
fmt.Println(now.Format("2006-01-02 03:04:05.000 PM Mon Jan"))
fmt.Println(now.Format("2006/01/02 15:04"))
fmt.Println(now.Format("15:04 2006/01/02"))
fmt.Println(now.Format("2006/01/02"))
}
解析字符串格式的时间
now := time.Now()
fmt.Println(now)
// 加载时区
loc, err := time.LoadLocation("Asia/Shanghai")
if err != nil {
fmt.Println(err)
return
}
// 按照指定时区和指定格式解析字符串时间
timeObj, err := time.ParseInLocation("2006/01/02 15:04:05", "2019/08/04 14:15:20", loc)
if err != nil {
fmt.Println(err)
return
}
fmt.Println(timeObj)
fmt.Println(timeObj.Sub(now))
缩写取前三位:
十二月份:January、February、March、April、May、June、July、August、September、October、November、December
一周:Monday、Tuesday、Wednesday、Thursday、Friday、Saturday、Sunday