# time包

# time.Tick() 定期循环

如果你从其它语言而来,要实现固定时间重复执行,第一个想到的代码可能是

for {
    time.Sleep(2 * time.Second)  // 2 秒间隔
  do_some_thing()
}

但是,更优雅的方案其实是

for range time.Tick(2 * time.Second ) {
  do_some_thing()
}

time.Tick() 函数返回的是一个 channel ,每隔指定的时间就会有数据从 channel 中出来。

range 其实是可以不接收遍历的值的,如果不接收,就可以简写成上面的格式。

那为什么我们不推荐 time.Sleep() 的方式呢,毕竟从上面的例子上看,感觉也没差多少啊?

但是,如果我们要指定退出机制,用 time.Sleep() 可能的做法就是

stop := false

go func(){
  for {
    if !stop {
      break;
    }
    do_some_thing()
    time.Sleep(2 * time.Second)  // 2 秒间隔
  }
}()

do_some_other_thing()
stop = false

而用 time.Tick() 的做法则是

func main() {
 t := time.NewTicker(time.Second * 2)

	go func(){
		for {
			select {
			case <- t.C:
				fmt.Println("内存循环")  // 只要不停止就会执行下面的函数
			}
			fmt.Println("外层循环")
		}
	}()

	time.Sleep(10*time.Second)
	t.Stop()
}

从逻辑和语义上来将 time.NewTicker() 的方式显然更 go

# 时间类型

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

# 时间戳

时间戳是自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)  // 1970-01-02 18:15:44 +0800 CST
	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)  // 1970-01-02 18:15:44
}

func main() {
	timestampDemo2(123344)
}

# 时间间隔

time.Durationtime包定义的一个类型,它代表两个时间点之间经过的时间,以纳秒为单位。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秒。

# 时间操作

# 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之后,返回真;否则返回假。

# 时间格式化

时间类型有一个自带的方法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"))
}

/*
2020-09-01 14:22:44.152 Tue Sep
2020-09-01 02:22:44.152 PM Tue Sep
2020/09/01 14:22
14:22 2020/09/01
2020/09/01
*/

# 解析字符串格式的时间

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))

/*
2020-09-01 14:23:36.910681 +0800 CST m=+0.000084741
2019-08-04 14:15:20 +0800 CST
-9456h8m16.910681s*/