這篇文章給大家介紹如何在golang中使用time包,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
成都創(chuàng)新互聯(lián)2013年至今,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目成都網(wǎng)站建設(shè)、網(wǎng)站建設(shè)網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元江達(dá)做網(wǎng)站,已為上家服務(wù),為江達(dá)各地企業(yè)和個人服務(wù),聯(lián)系電話:028-86922220
時間可分為時間點(diǎn)與時間段,golang 也不例外,提供了以下兩種基礎(chǔ)類型
- 時間點(diǎn)(Time)
- 時間段(Duration)
除此之外 golang 也提供了以下類型,做一些特定的業(yè)務(wù)
- 時區(qū)(Location)
- Ticker
- Timer(定時器)
我們將按以上順序來介紹 time 包的使用。
時間點(diǎn)(Time)
我們使用的所有與時間相關(guān)的業(yè)務(wù)都是基于點(diǎn)而延伸的,兩點(diǎn)組成一個時間段,大多數(shù)應(yīng)用也都是圍繞這些點(diǎn)與面去做邏輯處理。
初始化
go 針對不同的參數(shù)類型提供了以下初始化的方式
// func Now() Time fmt.Println(time.Now()) // func Parse(layout, value string) (Time, error) time.Parse("2016-01-02 15:04:05", "2018-04-23 12:24:51") // func ParseInLocation(layout, value string, loc *Location) (Time, error) (layout已帶時區(qū)時可直接用Parse) time.ParseInLocation("2006-01-02 15:04:05", "2017-05-11 14:06:06", time.Local) // func Unix(sec int64, nsec int64) Time time.Unix(1e9, 0) // func Date(year int, month Month, day, hour, min, sec, nsec int, loc *Location) Time time.Date(2018, 1, 2, 15, 30, 10, 0, time.Local) // func (t Time) In(loc *Location) Time 當(dāng)前時間對應(yīng)指定時區(qū)的時間 loc, _ := time.LoadLocation("America/Los_Angeles") fmt.Println(time.Now().In(loc)) // func (t Time) Local() Time
獲取到時間點(diǎn)之后為了滿足業(yè)務(wù)和設(shè)計,需要轉(zhuǎn)換成我們需要的格式,也就是所謂的時間格式化。
格式化
to string
格式化為字符串我們需要使用 time.Format 方法來轉(zhuǎn)換成我們想要的格式
fmt.Println(time.Now().Format("2006-01-02 15:04:05")) // 2018-04-24 10:11:20 fmt.Println(time.Now().Format(time.UnixDate)) // Tue Apr 24 09:59:02 CST 2018
Format 函數(shù)中可以指定你想使用的格式,同時 time 包中也給了一些我們常用的格式
const ( ANSIC = "Mon Jan _2 15:04:05 2006" UnixDate = "Mon Jan _2 15:04:05 MST 2006" RubyDate = "Mon Jan 02 15:04:05 -0700 2006" RFC822 = "02 Jan 06 15:04 MST" RFC822Z = "02 Jan 06 15:04 -0700" // RFC822 with numeric zone RFC850 = "Monday, 02-Jan-06 15:04:05 MST" RFC1123 = "Mon, 02 Jan 2006 15:04:05 MST" RFC1123Z = "Mon, 02 Jan 2006 15:04:05 -0700" // RFC1123 with numeric zone RFC3339 = "2006-01-02T15:04:05Z07:00" RFC3339Nano = "2006-01-02T15:04:05.999999999Z07:00" Kitchen = "3:04PM" // Handy time stamps. Stamp = "Jan _2 15:04:05" StampMilli = "Jan _2 15:04:05.000" StampMicro = "Jan _2 15:04:05.000000" StampNano = "Jan _2 15:04:05.000000000" )
注意: galang 中指定的特定時間格式為 "2006-01-02 15:04:05 -0700 MST", 為了記憶方便,按照美式時間格式 月日時分秒年 外加時區(qū) 排列起來依次是 01/02 03:04:05PM ‘06 -0700,剛開始使用時需要注意。
to time stamp
func (t Time) Unix() int64 func (t Time) UnixNano() int64 fmt.Println(time.Now().Unix()) // 獲取指定日期的時間戳 dt, _ := time.Parse("2016-01-02 15:04:05", "2018-04-23 12:24:51") fmt.Println(dt.Unix()) fmt.Println(time.Date(2018, 1,2,15,30,10,0, time.Local).Unix())
其他
time 包還提供了一些常用的方法,基本覆蓋了大多數(shù)業(yè)務(wù),從方法名就能知道代表的含義就不一一說明了。
func (t Time) Date() (year int, month Month, day int) func (t Time) Clock() (hour, min, sec int) func (t Time) Year() int func (t Time) Month() Month func (t Time) Day() int func (t Time) Hour() int func (t Time) Minute() int func (t Time) Second() int func (t Time) Nanosecond() int func (t Time) YearDay() int func (t Time) Weekday() Weekday func (t Time) ISOWeek() (year, week int) func (t Time) IsZero() bool func (t Time) Local() Time func (t Time) Location() *Location func (t Time) Zone() (name string, offset int) func (t Time) Unix() int64
時間段(Duartion)
介紹完了時間點(diǎn),我們再來介紹時間段,即 Duartion 類型, 我們業(yè)務(wù)也是很常用的類型。
// func ParseDuration(s string) (Duration, error) tp, _ := time.ParseDuration("1.5s") fmt.Println(tp.Truncate(1000), tp.Seconds(), tp.Nanoseconds()) func (d Duration) Hours() float64 func (d Duration) Minutes() float64 func (d Duration) Seconds() float64 func (d Duration) Nanoseconds() int64 func (d Duration) Round(m Duration) Duration // 四舍五入 func (d Duration) Truncate(m Duration) Duration // 向下取整
時區(qū)(Location)
我們在來介紹一下時區(qū)的相關(guān)的函數(shù)
// 默認(rèn)UTC loc, err := time.LoadLocation("") // 服務(wù)器設(shè)定的時區(qū),一般為CST loc, err := time.LoadLocation("Local") // 美國洛杉磯PDT loc, err := time.LoadLocation("America/Los_Angeles") // 獲取指定時區(qū)的時間點(diǎn) local, _ := time.LoadLocation("America/Los_Angeles") fmt.Println(time.Date(2018,1,1,12,0,0,0, local))
可以在 $GOROOT/lib/time/zoneinfo.zip 文件下看到所有時區(qū)。
時間運(yùn)算
好了,基礎(chǔ)的類型我們介紹完,現(xiàn)在開始時間運(yùn)算相關(guān)的函數(shù),也是日常業(yè)務(wù)中我們大量應(yīng)用的。
// func Sleep(d Duration) 休眠多少時間,休眠時處于阻塞狀態(tài),后續(xù)程序無法執(zhí)行 time.Sleep(time.Duration(10) * time.Second) // func After(d Duration) <-chan Time 非阻塞,可用于延遲 time.After(time.Duration(10) * time.Second) // func Since(t Time) Duration 兩個時間點(diǎn)的間隔 start := time.Now() fmt.Println(time.Since(start)) // 等價于 Now().Sub(t), 可用來計算一段業(yè)務(wù)的消耗時間 func Until(t Time) Duration // 等價于 t.Sub(Now()),t與當(dāng)前時間的間隔 // func (t Time) Add(d Duration) Time fmt.Println(dt.Add(time.Duration(10) * time.Second)) // 加 func (t Time) Sub(u Time) Duration // 減 // func (t Time) AddDate(years int, months int, days int) Time fmt.Println(dt.AddDate(1, 1, 1)) // func (t Time) Before(u Time) bool // func (t Time) After(u Time) bool // func (t Time) Equal(u Time) bool 比較時間點(diǎn)時盡量使用Equal函數(shù)
我們大概就介紹完了多數(shù)涉及時間點(diǎn)與時間段的函數(shù),接下面我們通過一些使用場景來做一些演示。
使用場景
日期時間差
dt1 := time.Date(2018, 1, 10, 0, 0, 1, 100, time.Local) dt2 := time.Date(2018, 1, 9, 23, 59, 22, 100, time.Local) // 不用關(guān)注時區(qū),go會轉(zhuǎn)換成時間戳進(jìn)行計算 fmt.Println(dt1.Sub(dt2))
基于當(dāng)前時間的前后運(yùn)算
now := time.Now() // 一年零一個月一天之后 fmt.Println(now.Date(1,1,1)) // 一段時間之后 fmt.Println(now.Add(time.Duration(10)*time.Minute)) // 計算兩個時間點(diǎn)的相差天數(shù) dt1 = time.Date(dt1.Year(), dt1.Month(), dt1.Day(), 0, 0, 0, 0, time.Local) dt2 = time.Date(dt2.Year(), dt2.Month(), dt2.Day(), 0, 0, 0, 0, time.Local) fmt.Println(int(math.Ceil(dt1.Sub(dt2).Hours() / 24)))
時區(qū)轉(zhuǎn)換
// time.Local 用來表示當(dāng)前服務(wù)器時區(qū) // 自定義地區(qū)時間 secondsEastOfUTC := int((8 * time.Hour).Seconds()) beijing := time.FixedZone("Beijing Time", secondsEastOfUTC) fmt.Println(time.Date(2018,1,2,0,0,0,0, beijing)) // 2018-01-02 00:00:00 +0800 Beijing Time // 當(dāng)前時間轉(zhuǎn)為指定時區(qū)時間 fmt.Println(time.Now().In(beijing)) // 指定時間轉(zhuǎn)換成指定時區(qū)對應(yīng)的時間 dt, err := time.ParseInLocation("2006-01-02 15:04:05", "2017-05-11 14:06:06", time.Local) // 當(dāng)前時間在零時區(qū)年月日 時分秒 時區(qū) year, mon, day := time.Now().UTC().Date() // 2018 April 24 hour, min, sec := time.Now().UTC().Clock() // 3 47 15 zone, _ := time.Now().UTC().Zone() // UTC
比較兩個時間點(diǎn)
dt := time.Date(2018, 1, 10, 0, 0, 1, 100, time.Local) fmt.Println(time.Now().After(dt)) // true fmt.Println(time.Now().Before(dt)) // false // 是否相等 判斷兩個時間點(diǎn)是否相等時推薦使用 Equal 函數(shù) fmt.Println(dt.Equal(time.Now()))
設(shè)置執(zhí)行時間
通過time.After 函數(shù)與 select 結(jié)合使用可用于處理程序超時設(shè)定
select { case m := <- c: // do something case <- time.After(time.Duration(1)*time.Second): fmt.Println("time out") }
Ticker類型
Ticker 類型包含一個 channel,有時我們會遇到每隔一段時間執(zhí)行的業(yè)務(wù)(比如設(shè)置心跳時間等),就可以用它來處理,這是一個重復(fù)的過程
// 無法取消 tick := time.Tick(1 * time.Minute) for _ = range tick { // do something } // 可通過調(diào)用ticker.Stop取消 ticker := time.NewTicker(1 * time.Minute) for _ = range tick { // do something }
Timer類型
Timer 類型用來代表一個單獨(dú)的事件,當(dāng)設(shè)置的時間過期后,發(fā)送當(dāng)前的時間到 channel, 我們可以通過以下兩種方式來創(chuàng)建
func AfterFunc(d Duration, f func()) *Timer // 指定一段時間后指定的函數(shù) func NewTimer(d Duration) *Timer
以上兩函數(shù)都可以使用 Reset, 這個有個需要注意的地方是使用 Reset 時需要確保 t.C 通道被釋放時才能調(diào)用,以防止發(fā)生資源競爭的問題,可通過以下方式解決
if !t.Stop() { <-t.C } t.Reset(d)
關(guān)于如何在golang中使用time包就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。