「有問必答」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))





請(qǐng)前往:http://lygongshang.com/TeacherV2.html?id=365