Go でカスタムの Time 型エイリアスを印刷すると予期しない出力が生成されるのはなぜですか?また、それを修正するにはどうすればよいですか?

Patricia Arquette
リリース: 2024-10-30 15:18:42
オリジナル
328 人が閲覧しました

Why does printing a custom Time type alias in Go produce unexpected output, and how can it be corrected?

Time.Time タイプ エイリアスを印刷するときに予期しない出力が表示される

Go では、カスタム Time タイプ エイリアスを印刷するときに予期しない出力が発生することがあります。動作を理解するには、問題をさらに詳しく分析する必要があります。

問題を理解する

次のコードを考えてみましょう:

<code class="go">package main
import (
    "encoding/json"
    "fmt"
    "strings"
    "time"
)

type Time time.Time

func (st *Time) UnmarshalJSON(b []byte) error {
    s := strings.Trim(string(b), "\"")
    t, err := time.Parse(time.RFC3339, fmt.Sprintf("%s%s", s, "Z"))
    if err != nil {
        return fmt.Errorf("parse time: %w", err)
    }
    *st = Time(t)
    return nil
}

type User struct {
    Name string
    TS Time
}

const data = `{id: 3, name: "Name", ts: "2021-05-21T03:10:20.958450"}`

func main() {
    user := new(User)
    json.Unmarshal([]byte(data), &user)
    fmt.Printf("%v\n", user)
}</code>
ログイン後にコピー

このコードを実行すると、予期される出力はフォーマットされた時間です。

&{Name 2021-05-21 03:10:20.95845 +0000 UTC}
ログイン後にコピー

のような値ですが、実際の出力は次のようになります。

&{Name {958450000 63757163420 <nil>}}
ログイン後にコピー

間違った出力の説明

Time 型のエイリアスが一致しないため、矛盾が生じます。 fmt.Stringer を実装すると、デフォルトの書式設定ロジックが引き継がれます。このロジックは、中括弧で囲まれた基になる time.Time 値のフィールドを出力します。

出力の修正

これを解決するには、time.Time.String にデリゲートする Time 型の String メソッドを実装します。 。これにより、目的の形式の出力が有効になります。

<code class="go">func (t Time) String() string {
    return time.Time(t).String()
}</code>
ログイン後にコピー

代替ソリューション

別のオプションは、Time タイプに time.Time を埋め込むことです。これにより、String メソッドとその他のメソッド (Marshal* を含む) が自動的にプロモートされます。

<code class="go">type Time struct {
    time.Time
}

func (st *Time) UnmarshalJSON(b []byte) error {
    // ... unchanged ...
    st.Time = t // simple assignment without type conversion
    // ... unchanged ...
}</code>
ログイン後にコピー

追加の考慮事項

JSON を解析するときは、文字列を使用した手動解析を避けてください。代わりに、適切なデコードのために json.Unmarshal を使用してください。さらに、time.ParseInLocation:

<code class="go">func (st *Time) UnmarshalJSON(b []byte) error {
    var s string
    if err := json.Unmarshal(b, &s); err != nil {
        return err
    }

    t, err := time.ParseInLocation("2006-01-02T15:04:05", s, time.UTC)
    if err != nil {
        return err
    }

    // ... unchanged ...
}</code>
ログイン後にコピー
を使用して時間解析を簡素化します。

以上がGo でカスタムの Time 型エイリアスを印刷すると予期しない出力が生成されるのはなぜですか?また、それを修正するにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
著者別の最新記事
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート
私たちについて 免責事項 Sitemap
PHP中国語ウェブサイト:福祉オンライン PHP トレーニング,PHP 学習者の迅速な成長を支援します!