乐趣区

关于go:有问必答Go如何优雅的对时间进行格式化

昨天 交换群 对于「Go 如何优雅的对工夫进行格式化?」开展了探讨:

咋搞捏?

如何在不循环的状况下,把列表数据结构体的工夫批改为咱们习惯的格局,而不是 UTC 模式

咱们要实现的成果如下:

  • created_at 是 go 语言原生的形式,
  • updated_at 是咱们冀望优化成的形式

    {
      "code": 200,
      "data": {
          "count": 12,
          "info": [
              {
                  "created_at": "2021-03-17T07:11:24+08:00" // 原生形式
                  "updated_at": "2021-03-17 07:11:24",  // 须要优化成这种
              }
          ]
      },
      "message": "胜利"
    }

引入神器

  1. 首先咱们引入一个包,在控制台运行

    go get github.com/liamylian/jsontime
  2. 下载相干依赖
go mod download
  1. 批改构造体,申明要解决工夫的字段

    type Order struct {
     .
     .
     .
     CreatedAt       time.Time `json:"created_at" time_format:"sql_datetime" time_utc:"false"`       // 格式化工夫示例
     UpdatedAt       string  `json:"updated_at"`       // 原生状态示例
    }
  2. 取值时调用 MarshalToString 把构造体数据转为字符串
  3. 然而转完的字符串存在反斜线的问题,应用 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,
         "胜利",
     })
    }

咱们最终实现进去的成果

{
    "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 包提供了工夫的显示和测量用的函数。日历的计算采纳的是公历。

工夫类型

time.Time 类型示意工夫。咱们能够通过 time.Now() 函数获取以后的工夫对象,而后获取工夫对象的年月日时分秒等信息。示例代码如下:

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)至以后工夫的总毫秒数。它也被称为 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() 函数能够将工夫戳转为工夫格局。

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

判断两个工夫是否雷同,会思考时区的影响,因而不同时区规范的工夫也能够正确比拟。本办法和用 t == u 不同,这种办法还会比拟地点和时区信息。

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)// 每秒都会执行的工作
    }
}

工夫格式化

工夫类型有一个自带的办法 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)
// 加载时区
loc, err := time.LoadLocation("Asia/Shanghai")
if err != nil {fmt.Println(err)
    return
}
// 依照指定时区和指定格局解析字符串工夫
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))

欢送关注 ❤

我的微信:wangzhongyang1993

视频号:王中阳 Go

公众号:程序员升职加薪之旅

退出移动版