<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
作為程式設計師,我們經常需要對時間進行處理。在 Go 中,標準庫 time 提供了對應的能力。
本文將介紹 time 庫中一些重要的函數和方法,希望能幫助到那些一遇到 Go 時間處理問題就需要百度的童鞋。
在程式設計中,我們經常會遭遇八小時時間差問題。這是由時區差異引起的,為了能更好地解決它們,我們需要理解幾個時間定義標準。
GMT(Greenwich Mean Time),格林威治平時。GMT 根據地球的自轉和公轉來計算時間,它規定太陽每天經過位於英國倫敦郊區的皇家格林威治天文臺的時間為中午12點。GMT 是前世界標準時。
UTC(Coordinated Universal Time),協調世界時。UTC 比 GMT 更精準,它根據原子鐘來計算時間。在不需要精確到秒的情況下,可以認為 UTC=GMT。UTC 是現世界標準時。
從格林威治本初子午線起,往東為正,往西為負,全球共劃分為 24 個標準時區,相鄰時區相差一個小時。
package main import ( "fmt" "time" ) func main() { fmt.Println(time.Now()) }
中國大陸使用的是東八時區的標準時,即北京時間 CST,China Standard Time。
$ go run main.go 2022-07-17 16:37:31.186043 +0800 CST m=+0.000066647
這是預設時區下的結果,time.Now()的列印中會標註+0800 CST。
假設我們是在美國洛杉磯時區下,那得到的結果是什麼呢?
$ TZ="America/Los_Angeles" go run main.go 2022-07-17 01:39:12.391505 -0700 PDT m=+0.000069514
可以看到,此時的結果是-0700 PDT 時間,即 PDT(Pacific Daylight Time)太平洋夏季時間。由於時區差異,兩次執行的時間結果相差了 15 小時。
注意,在使用 Docker 容器時,系統預設的時區就是 UTC 時間(0 時區),和我們實際需要的北京時間相差八個小時,這是導致八小時時間差問題的經典場景。
時區問題的應對策略,可以詳細檢視 src/time/zoneinfo_unix.go 中 initLocal() 函數的載入邏輯。例如,可以通過指定環境變數 TZ,修改/etc/localtime檔案等方式來解決。
因為時區問題非常重要,所以放在了文章第一部分講述。下面開始介紹 time 庫的使用。
time 庫,最核心的物件是 time.Time 結構體。它的定義如下,用以表示某個瞬間的時間。
type Time struct { // wall and ext encode the wall time seconds, wall time nanoseconds, // and optional monotonic clock reading in nanoseconds. wall uint64 ext int64 loc *Location }
計算機在時間處理上,主要涉及到兩種時鐘。
wall 和 ext 欄位就是用於記錄牆上時鐘和單調時鐘,精度為納秒。欄位的對應位數上關聯著用於確定時間的具體年、月、日、小時、分鐘、秒等資訊。
loc 欄位記錄時區位置,當 loc 為 nil 時,預設為 UTC 時間。
因為 time.Time 用於表示具有納秒精度的時間瞬間,在程式中通常應該將它作為值儲存和傳遞,而不是指標。
即在時間變數或者結構體欄位中,我們應該使用 time.Time,而非 *time.Time。
我們可以通過 Now 函數獲取當前本地時間
func Now() Time {}
也可以通過 Date 函數,根據年、月、日等時間和時區引數獲取指定時間
func Date(year int, month Month, day, hour, min, sec, nsec int, loc *Location) Time {}
計算機世界中,將 UTC 時間 1970 年1月1日 0 時 0 分 0 秒作為 Unix 時間 0。所謂的時間瞬間轉換為 Unix 時間戳,即計算的是從 Unix 時間 0 到指定瞬間所經過的秒數、微秒數等。
func (t Time) Unix() int64 {} // 從 Unix 時間 0 經過的秒數 func (t Time) UnixMicro() int64 {} // 從 Unix 時間 0 經過的微秒數 func (t Time) UnixMilli() int64 {} // 從 Unix 時間 0 經過的毫秒數 func (t Time) UnixNano() int64 {} // 從 Unix 時間 0 經過的納秒數
t := time.Now() fmt.Println(t.Date()) // 2022 July 17 fmt.Println(t.Year()) // 2022 fmt.Println(t.Month()) // July fmt.Println(t.ISOWeek()) // 2022 28 fmt.Println(t.Clock()) // 22 21 56 fmt.Println(t.Day()) // 17 fmt.Println(t.Weekday()) // Sunday fmt.Println(t.Hour()) // 22 fmt.Println(t.Minute()) // 21 fmt.Println(t.Second()) // 56 fmt.Println(t.Nanosecond())// 494313000 fmt.Println(t.YearDay()) // 198
持續時間 time.Duration 用於表示兩個時間瞬間 time.Time 之間所經過的時間。它通過 int64 表示納秒計數,能表示的極限大約為 290 年。
// A Duration represents the elapsed time between two instants // as an int64 nanosecond count. The representation limits the // largest representable duration to approximately 290 years. type Duration int64
在 Go 中,持續時間只是一個以納秒為單位的數位而已。如果持續時間等於 1000000000,則它代表的含義是 1 秒或 1000 毫秒或 1000000 微秒或 1000000000 納秒。
例如,相隔 1 小時的兩個時間瞬間 time.Time 值,它們之間的持續時間 time.Duration 值為
1*60*60*1000*1000*1000
Go 的 time 包中定義了這些持續時間常數值
const ( Nanosecond Duration = 1 Microsecond = 1000 * Nanosecond Millisecond = 1000 * Microsecond Second = 1000 * Millisecond Minute = 60 * Second Hour = 60 * Minute )
同時,time.Duration 提供了能獲取各時間粒度數值的方法
func (d Duration) Nanoseconds() int64 {} // 納秒 func (d Duration) Microseconds() int64 {} // 微秒 func (d Duration) Milliseconds() int64 {} // 毫秒 func (d Duration) Seconds() float64 {} // 秒 func (d Duration) Minutes() float64 {} // 分鐘 func (d Duration) Hours() float64 {} // 小時
在學習了時間瞬間和持續時間之後,我們來看如何做時間計算。
func (t Time) Add(d Duration) Time {}
Add 函數用於增加/減少( d 的正值表示增加、負值表示減少) time.Time 的持續時間。我們可以對某瞬時時間,增加或減少指定納秒級以上的時間。
func (t Time) Sub(u Time) Duration {}
Sub 函數可以得出兩個時間瞬間之間的持續時間。
func (t Time) AddDate(years int, months int, days int) Time {}
AddDate 函數基於年、月和日的維度增加/減少 time.Time 的值。
當然,基於當前時間瞬間 time.Now() 的計算是最普遍的需求。因此,time 包還提供了以下便捷的時間計算函數。
func Since(t Time) Duration {}
Since 函數是 time.Now().Sub(t) 的快捷方法。
func Until(t Time) Duration {}
Until 函數是 t.Sub(time.Now()) 的快捷方法。
t := time.Now() fmt.Println(t) // 2022-07-17 22:41:06.001567 +0800 CST m=+0.000057466 //時間增加 1小時 fmt.Println(t.Add(time.Hour * 1)) // 2022-07-17 23:41:06.001567 +0800 CST m=+3600.000057466 //時間增加 15 分鐘 fmt.Println(t.Add(time.Minute * 15))// 2022-07-17 22:56:06.001567 +0800 CST m=+900.000057466 //時間增加 10 秒鐘 fmt.Println(t.Add(time.Second * 10))// 2022-07-17 22:41:16.001567 +0800 CST m=+10.000057466 //時間減少 1 小時 fmt.Println(t.Add(-time.Hour * 1)) // 2022-07-17 21:41:06.001567 +0800 CST m=-3599.999942534 //時間減少 15 分鐘 fmt.Println(t.Add(-time.Minute * 15))// 2022-07-17 22:26:06.001567 +0800 CST m=-899.999942534 //時間減少 10 秒鐘 fmt.Println(t.Add(-time.Second * 10))// 2022-07-17 22:40:56.001567 +0800 CST m=-9.999942534 time.Sleep(time.Second * 5) t2 := time.Now() // 計算 t 到 t2 的持續時間 fmt.Println(t2.Sub(t)) // 5.004318874s // 1 年之後的時間 t3 := t2.AddDate(1, 0, 0) // 計算從 t 到當前的持續時間 fmt.Println(time.Since(t)) // 5.004442316s // 計算現在到明年的持續時間 fmt.Println(time.Until(t3)) // 8759h59m59.999864s
在其他語言中,一般會使用通用的時間模板來格式化時間。例如 Python,它使用 %Y 代表年、%m 代表月、%d 代表日等。
但是,Go 不一樣,它使用固定的時間(需要注意,使用其他的時間是不可以的)作為佈局模板,而這個固定時間是 Go 語言的誕生時間。
Mon Jan 2 15:04:05 MST 2006
格式化時間涉及到兩個轉換函數
func Parse(layout, value string) (Time, error) {}
Parse 函數用於將時間字串根據它所能對應的佈局轉換為 time.Time 物件。
func (t Time) Format(layout string) string {}
Formate 函數用於將 time.Time 物件根據給定的佈局轉換為時間字串。
範例
const ( layoutISO = "2006-01-02" layoutUS = "January 2, 2006" ) date := "2012-08-09" t, _ := time.Parse(layoutISO, date) fmt.Println(t) // 2012-08-09 00:00:00 +0000 UTC fmt.Println(t.Format(layoutUS)) // August 9, 2012
在 time 庫中,Go 提供了一些預定義的佈局模板常數,這些可以直接拿來使用。
const ( Layout = "01/02 03:04:05PM '06 -0700" // The reference time, in numerical order. 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" )
下面是我們可選的佈局引數對照表
年 06/2006 月 01/1/Jan/January 日 02/2/_2 星期 Mon/Monday 小時 03/3/15 分 04/4 秒 05/5 毫秒 .000/.999 微秒 .000000/.999999 納秒 .000000000/.999999999 am/pm PM/pm 時區 MST 時區小時數差-0700/-07/-07:00/Z0700/Z07:00
在文章開頭,我們介紹了時區問題。如果在程式碼中,需要獲取同一個 time.Time 在不同時區下的結果,我們可以使用它的 In 方法。
func (t Time) In(loc *Location) Time {}
它的使用非常簡單,直接看範例程式碼
now := time.Now() fmt.Println(now) // 2022-07-18 21:19:59.9636 +0800 CST m=+0.000069242 loc, _ := time.LoadLocation("UTC") fmt.Println(now.In(loc)) // 2022-07-18 13:19:59.9636 +0000 UTC loc, _ = time.LoadLocation("Europe/Berlin") fmt.Println(now.In(loc)) // 2022-07-18 15:19:59.9636 +0200 CEST loc, _ = time.LoadLocation("America/New_York") fmt.Println(now.In(loc)) // 2022-07-18 09:19:59.9636 -0400 EDT loc, _ = time.LoadLocation("Asia/Dubai") fmt.Println(now.In(loc)) // 2022-07-18 17:19:59.9636 +0400 +04
整體而言,time 庫提供的時間處理常式和方法,基本滿足我們的使用需求。
有意思的是,Go 時間格式化轉換必須採用 Go 誕生時間,確實有夠自戀。
到此這篇關於詳解Go 中的時間處理的文章就介紹到這了,更多相關go時間處理內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45