Go時(shí)間格式化的實(shí)現(xià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": "成功"
}引入神器
首先我們引入一個包,在控制臺運(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": "成功"
}好了,通過引入上面的神器就解決了我們的問題。
我們再深入理解一下time包的使用:
time包
time包提供了時(shí)間的顯示和測量用的函數(shù)。日歷的計(jì)算采用的是公歷。
時(shí)間類型
time.Time類型表示時(shí)間。我們可以通過time.Now()函數(shù)獲取當(dāng)前的時(shí)間對象,然后獲取時(shí)間對象的年月日時(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í)間對象獲取時(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包定義的一個類型,它代表兩個時(shí)間點(diǎn)之間經(jīng)過的時(shí)間,以納秒為單位。time.Duration表示一段時(shí)間間隔,可表示的最長時(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
我們在日常的編碼過程中可能會遇到要求時(shí)間+時(shí)間間隔的需求,Go語言的時(shí)間對象有提供Add方法如下:
func (t Time) Add(d Duration) Time
舉個例子,求一個小時(shí)之后的時(shí)間:
func main() {
now := time.Now()
later := now.Add(time.Hour) // 當(dāng)前時(shí)間加1小時(shí)后的時(shí)間
fmt.Println(later)
}Sub
求兩個時(shí)間之間的差值:
func (t Time) Sub(u Time) Duration
返回一個時(shí)間段t-u。如果結(jié)果超出了Duration可以表示的最大值/最小值,將返回最大值/最小值。要獲取時(shí)間點(diǎn)t-d(d為Duration),可以使用t.Add(-d)。
Equal
func (t Time) Equal(u Time) bool
判斷兩個時(shí)間是否相同,會考慮時(shí)區(qū)的影響,因此不同時(shí)區(qū)標(biāo)準(zhǔn)的時(shí)間也可以正確比較。本方法和用t==u不同,這種方法還會比較地點(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ì)上是一個通道(channel)。
func tickDemo() {
ticker := time.Tick(time.Second) //定義一個1秒間隔的定時(shí)器
for i := range ticker {
fmt.Println(i)//每秒都會執(zhí)行的任務(wù)
}
}時(shí)間格式化
時(shí)間類型有一個自帶的方法Format進(jìn)行格式化,需要注意的是Go語言中格式化時(shí)間模板不是常見的Y-m-d H:M:S而是使用Go的誕生時(shí)間2006年1月2號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號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))到此這篇關(guān)于Go時(shí)間格式化的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Go時(shí)間格式化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言數(shù)據(jù)結(jié)構(gòu)之希爾排序示例詳解
Go語言時(shí)間相關(guān)操作合集(超詳細(xì))

