Go日志之日志庫講解

1 Go日志

1.1 引言

Go語言是一種高性能、簡潔、并發(fā)友好的編程語言,廣泛用于開發(fā)各種應(yīng)用程序,如網(wǎng)絡(luò)服務(wù)、微服務(wù)、云計算、區(qū)塊鏈等。Go語言的標(biāo)準(zhǔn)庫提供了一個基本的日志包(log),可以用于記錄簡單的日志信息,如時間、級別、消息等。但是,如果需要更多的功能和靈活性,例如結(jié)構(gòu)化日志、日志旋轉(zhuǎn)、日志鉤子、日志格式化等,那么可能需要使用第三方的日志庫。

本文將介紹Go語言中最流行和最優(yōu)秀的日志庫,并給出每個庫的使用示例代碼

1.2 Zap

Zap 是一個快速、結(jié)構(gòu)化、分級和可擴展的日志庫,由Uber開發(fā)和維護。Zap提供了兩種不同的APISugaredLoggerLoggerSugaredLogger支持結(jié)構(gòu)化和非結(jié)構(gòu)化的日志記錄,但是犧牲了一些性能。Logger只支持結(jié)構(gòu)化的日志記錄,但是具有更高的性能和更低的內(nèi)存分配。

Zap還提供了一些有用的特性,如:

  • 可以自定義日志級別、輸出目標(biāo)、編碼器(JSON或控制臺)、時間格式等。
  • 可以使用字段(Fields)來添加結(jié)構(gòu)化的上下文信息,如鍵值對。
  • 可以使用鉤子(Hooks)來在每次寫入日志時執(zhí)行一些操作,如發(fā)送郵件、寫入數(shù)據(jù)庫等。
  • 可以使用取樣器(Sampler)來限制每秒寫入的日志數(shù)量,以減少性能開銷。
  • 可以使用核心(Core)來組合多個輸出目標(biāo)和編碼器。

以下是一個使用Zap的示例代碼:

package main

import (
 "go.uber.org/zap"
)

func main() {
 // 創(chuàng)建一個 Logger
 logger, err := zap.NewDevelopment()
 if err != nil {
  panic(err)
 }
 defer logger.Sync()
 sugar := logger.Sugar()

 // 使用SugaredLogger記錄非結(jié)構(gòu)化的日志
 sugar.Infow("This is a structured log",
  "key1", "value1",
  "key2", "value2",
 )

 // 使用SugaredLogger記錄結(jié)構(gòu)化的日志
 sugar.Infof("This is an unstructured log: %s", "hello world")

 // 創(chuàng)建一個Logger
 logger, err = zap.NewProduction()
 if err != nil {
  panic(err)
 }
 defer logger.Sync()

 // 使用Logger記錄結(jié)構(gòu)化的日志
 logger.Info("This is a structured log",
  zap.String("key1", "value1"),
  zap.String("key2", "value2"),
 )

 // 使用Logger記錄錯誤級別的日志,并添加堆棧跟蹤信息
 logger.Error("This is an error log",
  zap.Error(err),
  zap.Stack("stack"),
 )
}

1.3 log/slog

log/slogGo 1.21 中引入的一個新的結(jié)構(gòu)化日志庫,它與標(biāo)準(zhǔn)庫的log包兼容,但提供了更多的功能和靈活性。log/slog定義了一個類型 Logger,用于記錄不同級別和格式的日志信息。每個 Logger 都關(guān)聯(lián)一個Handler,用于處理日志記錄。log/slog還提供了一個默認的 Logger,可以通過頂級函數(shù)(如Info和Error)來使用,它們會調(diào)用相應(yīng)的Logger方法。該默認Logger將日志信息寫入標(biāo)準(zhǔn)錯誤,并在每條日志信息前添加日期和時間。

log/slog的日志記錄由以下幾個部分組成:

  • 時間:日志記錄發(fā)生的時間,可以是本地時間或UTC時間。
  • 級別:日志記錄的嚴重程度,可以是預(yù)定義的四個級別之一(Debug、Info、Warn、Error),也可以是自定義的整數(shù)值。
  • 消息:日志記錄的主要內(nèi)容,通常是一個簡短的描述性字符串。
  • 屬性:日志記錄的額外信息,以鍵值對的形式表示,鍵是字符串,值可以是任意類型。

例如,以下代碼:

package main

import (
 "log/slog"
 "os"
)

func main() {
 slog.Info("hello, world", "user", os.Getenv("USER"))
}

會產(chǎn)生以下輸出:

2023/09/09 16:27:19 INFO hello, world user=polarisxu

其中,2023/09/09 16:27:19是時間,INFO是級別,hello, world是消息,user=polarisxu是屬性。

log/slog還提供了一些有用的特性,如:

  • 可以自定義日志級別、輸出目標(biāo)、格式器(JSON或文本)、時間戳等。
  • 可以使用字段(Fields)來添加結(jié)構(gòu)化的上下文信息,如鍵值對。
  • 可以使用處理器(Handler)來處理不同級別或條件的日志信息,如過濾、分割、彩色等。
    v可以使用條目(Entry)來記錄帶有字段的日志信息,或者使用WithFields、WithTime、WithError等方法來創(chuàng)建帶有字段的條目。
  • 可以使用日志級別函數(shù)(如Info、Warn、Error等)來記錄不同級別的日志信息,或者使用Log或Print等方法來記錄默認級別的日志信息。

以下是一個使用log/slog的示例代碼:

package main

import (
 "log/slog"
 "os"
)

func main() {
 // 創(chuàng)建一個JSON處理器
 jsonHandler := slog.NewJSONHandler(os.Stdout, nil)

 // 創(chuàng)建一個文本處理器
 textHandler := slog.NewTextHandler(os.Stderr, nil)

 // 創(chuàng)建一個文本 Logger
 textLogger := slog.New(textHandler)

 // 創(chuàng)建一個 JSON Logger
 jsonLogger := slog.New(jsonHandler)

 // 使用Logger記錄結(jié)構(gòu)化的日志信息
 textLogger.Info("hello, world", "user", os.Getenv("USER"))

 // 使用Logger記錄結(jié)構(gòu)化的日志信息
 jsonLogger.Info("hello, world", "user", os.Getenv("USER"))
}

該程序會在標(biāo)準(zhǔn)錯誤上輸出文本格式的日志信息:

time=2023-09-09T16:27:19.000-05:00 level=INFO msg=hello, world user=polarisxu

然后在標(biāo)準(zhǔn)輸出上輸出JSON格式的日志信息:

{"time":"2023-09-09T16:27:19.000000000-05:00","level":"INFO","msg":"hello","user":"polarisxu"}

1.4 Logrus

Logrus是一個結(jié)構(gòu)化、分級、可擴展和兼容標(biāo)準(zhǔn)庫log包的日志庫,由Sirupsen開發(fā)和維護。Logrus提供了一個簡單而強大的API,可以用于記錄不同級別和格式的日志信息。

Logrus也提供了一些有用的特性,如:

  • 可以自定義日志級別、輸出目標(biāo)、格式器(JSON或文本)、時間戳等。
  • 可以使用字段(Fields)來添加結(jié)構(gòu)化的上下文信息,如鍵值對。
  • 可以使用鉤子(Hooks)來在每次寫入日志時執(zhí)行一些操作,如發(fā)送郵件、寫入數(shù)據(jù)庫等。
  • 可以使用條目(Entry)來記錄帶有字段的日志信息,或者使用WithFields、WithTime、WithError等方法來創(chuàng)建帶有字段的條目。
  • 可以使用日志級別函數(shù)(如Info、Warn、Error等)來記錄不同級別的日志信息,或者使用Log或Print等方法來記錄默認級別的日志信息。

以下是一個使用Logrus的示例代碼:

package main

import (
 "os"

 "github.com/sirupsen/logrus"
)

func main() {
 // 創(chuàng)建一個Logrus實例
 log := logrus.New()

 // 設(shè)置日志級別為Debug
 log.SetLevel(logrus.DebugLevel)

 // 設(shè)置輸出目標(biāo)為標(biāo)準(zhǔn)輸出
 log.SetOutput(os.Stdout)

 // 設(shè)置格式器為JSON
 log.SetFormatter(&logrus.JSONFormatter{})

 // 使用Fields添加結(jié)構(gòu)化的上下文信息
 log.WithFields(logrus.Fields{
  "key1": "value1",
  "key2": "value2",
 }).Info("This is a structured log")

 // 使用Entry記錄帶有字段的日志信息
 entry := log.WithFields(logrus.Fields{
  "key3": "value3",
  "key4": "value4",
 })
 entry.Warn("This is another structured log")

 // 使用日志級別函數(shù)記錄不同級別的日志信息
 log.Debug("This is a debug log")
 log.Info("This is an info log")
 log.Warn("This is a warn log")
 log.Error("This is an error log")
 log.Fatal("This is a fatal log")
 log.Panic("This is a panic log")

 // 使用Log或Print等方法記錄默認級別的日志信息
 log.Log(logrus.InfoLevel, "This is a log with level")
 log.Print("This is a print log")
}

1.5 Zerolog

Zerolog是一個快速、簡單、零內(nèi)存分配的結(jié)構(gòu)化日志庫,由rs開發(fā)和維護。Zerolog 提供了一個流式(Fluent)的API,可以用于記錄不同級別和格式的日志信息。

Zerolog也提供了一些有用的特性,如:

  • 可以自定義日志級別、輸出目標(biāo)、編碼器(JSON或控制臺)、時間格式等。
  • 可以使用字段(Fields)來添加結(jié)構(gòu)化的上下文信息,如鍵值對。
  • 可以使用鉤子(Hooks)來在每次寫入日志時執(zhí)行一些操作,如發(fā)送郵件、寫入數(shù)據(jù)庫等。
  • 可以使用取樣器(Sampler)來限制每秒寫入的日志數(shù)量,以減少性能開銷。
  • 可以使用上下文(Context)來創(chuàng)建帶有字段的日志記錄器,或者使用With、Dict等方法來添加字段。

以下是一個使用Zerolog的示例代碼:

package main

import (
 "os"

 "github.com/rs/zerolog"
 "github.com/rs/zerolog/log"
)

func main() {
 // 設(shè)置日志級別為Debug
 zerolog.SetGlobalLevel(zerolog.DebugLevel)

 // 設(shè)置輸出目標(biāo)為標(biāo)準(zhǔn)輸出
 log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stdout})

 // 使用Fields添加結(jié)構(gòu)化的上下文信息
 log.Info().
  Str("key1", "value1").
  Str("key2", "value2").
  Msg("This is a structured log")

 // 使用Context創(chuàng)建帶有字段的日志記錄器
 sublogger := log.With().
  Str("key3", "value3").
  Logger()

 sublogger.Warn().
  Str("key4", "value4").
  Msg("This is another structured log")

 // 使用日志級別函數(shù)記錄不同級別的日志信息
 log.Debug().Msg("This is a debug log")
 log.Info().Msg("This is an info log")
 log.Warn().Msg("This is a warn log")
 log.Error().Msg("This is an error log")
 log.Fatal().Msg("This is a fatal log")
 log.Panic().Msg("This is a panic log")
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容