昨天 交换群 对于「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": "胜利" }
引入神器
-
首先咱们引入一个包,在控制台运行
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"` // 原生状态示例 }
- 取值时调用MarshalToString把构造体数据转为字符串
-
然而转完的字符串存在反斜线的问题,应用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
公众号:程序员升职加薪之旅
发表回复