「質問と回答」 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
  1. 関連する依存関係をダウンロードする
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"`       // 原生状态示例
}
  1. 値を取得するときに、MarshalToString を呼び出して構造データを文字列に変換します。
  2. ただし、変換された文字列にはバックスラッシュの問題があります。それに対処するには 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.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 パッケージによって定義されるタイプで、2 つの時点間の経過時間をナノ秒単位で表します。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 言語の time オブジェクトには次のような Add メソッドが用意されています。

func (t Time) Add(d Duration) Time
たとえば、1 時間後の時刻を求めます。

func main() {
    
    
	now := time.Now()
	later := now.Add(time.Hour) // 当前时间加1小时后的时间
	fmt.Println(later)
}

サブ

2 つの時間の差を求めます。

func (t Time) Sub(u Time) Duration

期間 tu を返します。結果がDurationが表現できる最大値/最小値を超える場合は、最大値/最小値が返されます。時点 td (d は期間) を取得するには、t.Add(-d) を使用できます。

同等

func (t Time) Equal(u Time) bool

2つの時刻が同じかどうかの判定にはタイムゾーンの影響が考慮されるため、異なるタイムゾーン基準の時刻も正確に比較できます。この方法は t==u を使用する場合とは異なり、位置情報とタイムゾーン情報も比較します。

func (t Time) Before(u Time) bool

t で表される時間が u より前の場合は true を返し、それ以外の場合は false を返します。

func (t Time) After(u Time) bool

t で表される時間が u より後の場合は true を返し、それ以外の場合は false を返します。

タイマー

time.Tick (時間間隔) を使用してタイマーを設定します。タイマーは基本的にチャネル (チャネル) です。

func tickDemo() {
    
    
	ticker := time.Tick(time.Second) //定义一个1秒间隔的定时器
	for i := range ticker {
    
    
		fmt.Println(i)//每秒都会执行的任务
	}
}

時間の書式設定

時刻タイプには書式設定のための組み込みメソッド Format があります。Go 言語で書式設定された時刻テンプレートは一般的な Ymd H:M:S ではなく、Go の誕生時刻である 1 月 2 日 15:04 を使用することに注意してください。 , 2006 (記憶式は 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))

フォロー大歓迎です❤

私のWeChat: wangzhongyang1993

ビデオID: Wang Zhongyang Go

公式アカウント:プログラマー昇進と昇給の旅

おすすめ

転載: blog.csdn.net/w425772719/article/details/131242001