「有問必答」Gopher如何優(yōu)雅的對時間進行格式化?
如何在不循環(huán)的情況下,把列表數(shù)據(jù)結構體的時間修改為咱們習慣的格式,而不是UTC模式
我們要實現(xiàn)的效果如下:
created_at 是go語言原生的方式,
updated_at 是我們期望優(yōu)化成的方式
{
"code": 200,
"data": {
"count": 12,
"info": [
{
"created_at": "2021-03-17T07:11:24+08:00" //原生方式
"updated_at": "2021-03-17 07:11:24", //需要優(yōu)化成這種
}
]
},
"message": "成功"
}
引入神器
首先我們引入一個包,在控制臺運行
go get github.com/liamylian/jsontime
下載相關依賴
go mod download
修改結構體,聲明要處理的時間字段
type Order struct {
.
.
.
CreatedAt time.Time `json:"created_at" time_format:"sql_datetime" time_utc:"false"` // 格式化時間示例
UpdatedAt string `json:"updated_at"` // 原生狀態(tài)示例
}
取值時調用MarshalToString把結構體數(shù)據(jù)轉為字符串
但是轉完的字符串存在反斜線的問題,使用json.RawMessage()處理一下
var timeJson = jsontime.ConfigWithCustomTimeFormat
func AllOrder(c *gin.Context) {
limitStr := c.DefaultQuery("limit", "10")
pageStr := c.DefaultQuery("page", "0")
orderType := c.DefaultQuery("orderType", "desc")
orderField := c.DefaultQuery("orderField", "id")
orderSql := orderField + " " + orderType
limit, _ := strconv.Atoi(limitStr)
page, _ := strconv.Atoi(pageStr)
count, res := model.QueryOrder(0, limit, page, orderSql)
//處理1:MarshalToString
bytes, _ := timeJson.MarshalToString(&res)
jsonInfo := map[string]interface{}{
"count": count,
//處理2:解決反斜線的問題
"info": json.RawMessage(bytes),
}
c.JSON(http.StatusOK, ReturnJson{
http.StatusOK,
jsonInfo,
"成功",
})
}
我們最終實現(xiàn)出來的效果
{
"code": 200,
"data": {
"count": 12,
"info": [
{
"updated_at": "2021-03-17 07:13:24",
"created_at": "2021-03-17 07:11:24",
}
]
},
"message": "成功"
}
好了,通過引入上面的神器就解決了我們的問題。
我們再深入理解一下time包的使用:
time包
time包提供了時間的顯示和測量用的函數(shù)。日歷的計算采用的是公歷。
時間類型
time.Time類型表示時間。我們可以通過time.Now()函數(shù)獲取當前的時間對象,然后獲取時間對象的年月日時分秒等信息。示例代碼如下:
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)至當前時間的總毫秒數(shù)。它也被稱為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()函數(shù)可以將時間戳轉為時間格式。
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)
}
時間間隔
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秒。
時間操作
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
判斷兩個時間是否相同,會考慮時區(qū)的影響,因此不同時區(qū)標準的時間也可以正確比較。本方法和用t==u不同,這種方法還會比較地點和時區(qū)信息。
Before
func (t Time) Before(u Time) bool
如果t代表的時間點在u之前,返回真;否則返回假。
After
func (t Time) After(u Time) bool
如果t代表的時間點在u之后,返回真;否則返回假。
定時器
使用time.Tick(時間間隔)來設置定時器,定時器的本質上是一個通道(channel)。
func tickDemo() {
ticker := time.Tick(time.Second) //定義一個1秒間隔的定時器
for i := range ticker {
fmt.Println(i)//每秒都會執(zhí)行的任務
}
}
時間格式化
時間類型有一個自帶的方法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)
// 加載時區(qū)
loc, err := time.LoadLocation("Asia/Shanghai")
if err != nil {
fmt.Println(err)
return
}
// 按照指定時區(qū)和指定格式解析字符串時間
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))