「有問必答」Gopher如何優(yōu)雅的對(duì)時(shí)間進(jìn)行格式化?
如何在不循環(huán)的情況下,把列表數(shù)據(jù)結(jié)構(gòu)體的時(shí)間修改為咱們習(xí)慣的格式,而不是UTC模式
我們要實(shí)現(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": "成功"
}
引入神器
首先我們引入一個(gè)包,在控制臺(tái)運(yùn)行
go get github.com/liamylian/jsontime
下載相關(guān)依賴
go mod download
修改結(jié)構(gòu)體,聲明要處理的時(shí)間字段
type Order struct {
.
.
.
CreatedAt time.Time `json:"created_at" time_format:"sql_datetime" time_utc:"false"` // 格式化時(shí)間示例
UpdatedAt string `json:"updated_at"` // 原生狀態(tài)示例
}
取值時(shí)調(diào)用MarshalToString把結(jié)構(gòu)體數(shù)據(jù)轉(zhuǎn)為字符串
但是轉(zhuǎn)完的字符串存在反斜線的問題,使用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,
"成功",
})
}
我們最終實(shí)現(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": "成功"
}
好了,通過引入上面的神器就解決了我們的問題。
我們?cè)偕钊肜斫庖幌聇ime包的使用:
time包
time包提供了時(shí)間的顯示和測(cè)量用的函數(shù)。日歷的計(jì)算采用的是公歷。
時(shí)間類型
time.Time類型表示時(shí)間。我們可以通過time.Now()函數(shù)獲取當(dāng)前的時(shí)間對(duì)象,然后獲取時(shí)間對(duì)象的年月日時(shí)分秒等信息。示例代碼如下:
func timeDemo() {
now := time.Now() //獲取當(dāng)前時(shí)間
fmt.Printf("current time:%v\n", now)
year := now.Year() //年
month := now.Month() //月
day := now.Day() //日
hour := now.Hour() //小時(shí)
minute := now.Minute() //分鐘
second := now.Second() //秒
fmt.Printf("%d-%02d-%02d %02d:%02d:%02d\n", year, month, day, hour, minute, second)
}
時(shí)間戳
時(shí)間戳是自1970年1月1日(08:00:00GMT)至當(dāng)前時(shí)間的總毫秒數(shù)。它也被稱為Unix時(shí)間戳(UnixTimestamp)。
基于時(shí)間對(duì)象獲取時(shí)間戳的示例代碼如下:
func timestampDemo() {
now := time.Now() //獲取當(dāng)前時(shí)間
timestamp1 := now.Unix() //時(shí)間戳
timestamp2 := now.UnixNano() //納秒時(shí)間戳
fmt.Printf("current timestamp1:%v\n", timestamp1)
fmt.Printf("current timestamp2:%v\n", timestamp2)
}
使用time.Unix()函數(shù)可以將時(shí)間戳轉(zhuǎn)為時(shí)間格式。
func timestampDemo2(timestamp int64) {
timeObj := time.Unix(timestamp, 0) //將時(shí)間戳轉(zhuǎn)為時(shí)間格式
fmt.Println(timeObj)
year := timeObj.Year() //年
month := timeObj.Month() //月
day := timeObj.Day() //日
hour := timeObj.Hour() //小時(shí)
minute := timeObj.Minute() //分鐘
second := timeObj.Second() //秒
fmt.Printf("%d-%02d-%02d %02d:%02d:%02d\n", year, month, day, hour, minute, second)
}
時(shí)間間隔
time.Duration是time包定義的一個(gè)類型,它代表兩個(gè)時(shí)間點(diǎn)之間經(jīng)過的時(shí)間,以納秒為單位。time.Duration表示一段時(shí)間間隔,可表示的最長(zhǎng)時(shí)間段大約290年。
time包中定義的時(shí)間間隔類型的常量如下:
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秒。
時(shí)間操作
Add 我們?cè)谌粘5木幋a過程中可能會(huì)遇到要求時(shí)間+時(shí)間間隔的需求,Go語言的時(shí)間對(duì)象有提供Add方法如下:
func (t Time) Add(d Duration) Time 舉個(gè)例子,求一個(gè)小時(shí)之后的時(shí)間:
func main() {
now := time.Now()
later := now.Add(time.Hour) // 當(dāng)前時(shí)間加1小時(shí)后的時(shí)間
fmt.Println(later)
}
Sub
求兩個(gè)時(shí)間之間的差值:
func (t Time) Sub(u Time) Duration
返回一個(gè)時(shí)間段t-u。如果結(jié)果超出了Duration可以表示的最大值/最小值,將返回最大值/最小值。要獲取時(shí)間點(diǎn)t-d(d為Duration),可以使用t.Add(-d)。
Equal
func (t Time) Equal(u Time) bool
判斷兩個(gè)時(shí)間是否相同,會(huì)考慮時(shí)區(qū)的影響,因此不同時(shí)區(qū)標(biāo)準(zhǔn)的時(shí)間也可以正確比較。本方法和用t==u不同,這種方法還會(huì)比較地點(diǎn)和時(shí)區(qū)信息。
Before
func (t Time) Before(u Time) bool
如果t代表的時(shí)間點(diǎn)在u之前,返回真;否則返回假。
After
func (t Time) After(u Time) bool
如果t代表的時(shí)間點(diǎn)在u之后,返回真;否則返回假。
定時(shí)器
使用time.Tick(時(shí)間間隔)來設(shè)置定時(shí)器,定時(shí)器的本質(zhì)上是一個(gè)通道(channel)。
func tickDemo() {
ticker := time.Tick(time.Second) //定義一個(gè)1秒間隔的定時(shí)器
for i := range ticker {
fmt.Println(i)//每秒都會(huì)執(zhí)行的任務(wù)
}
}
時(shí)間格式化
時(shí)間類型有一個(gè)自帶的方法Format進(jìn)行格式化,需要注意的是Go語言中格式化時(shí)間模板不是常見的Y-m-d H:M:S而是使用Go的誕生時(shí)間2006年1月2號(hào)15點(diǎn)04分(記憶口訣為2006 1 2 3 4)。
也許這就是技術(shù)人員的浪漫吧~(當(dāng)然,也有人說這是瞎搞~)
補(bǔ)充:如果想格式化為12小時(shí)方式,需指定PM。
func formatDemo() {
now := time.Now()
// 格式化的模板為Go的出生時(shí)間2006年1月2號(hào)15點(diǎn)04分 Mon Jan
// 24小時(shí)制
fmt.Println(now.Format("2006-01-02 15:04:05.000 Mon Jan"))
// 12小時(shí)制
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"))
}
解析字符串格式的時(shí)間
now := time.Now()
fmt.Println(now)
// 加載時(shí)區(qū)
loc, err := time.LoadLocation("Asia/Shanghai")
if err != nil {
fmt.Println(err)
return
}
// 按照指定時(shí)區(qū)和指定格式解析字符串時(shí)間
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))