go1.21中slog日志包用法入門
WHY
在日志處理上,我們從前使用的log包缺乏結構化的輸出,導致信息呈現(xiàn)出來的樣子并非最適合人類閱讀,而slog是一種結構化的日志,它可以用鍵值對的形式將我們需要的信息呈現(xiàn)出來,使得處理與分析日志變得更為容易。
HOW
1. 快速入門
package main import ( "log/slog" ) func main() { slog.Info("my first slog msg", "greeting", "hello, slog") slog.Error("my secod slog message", "greeting", "hello slog") slog.Warn("my third message", "greeting", "hello slog") }
以上是三條最簡單的slog語句,其結果是這樣的:
2023/09/10 21:51:03 INFO my first slog msg greeting="hello, slog"
2023/09/10 21:51:03 ERROR my secod slog message greeting="hello slog"
2023/09/10 21:51:03 WARN my third message greeting="hello slog"
這三行代碼中的第一個參數(shù)代表了log的message,我們可以看到,此時打印出來的日志信息是文本信息,那如何使得日志以非純文本(比如json)展現(xiàn)呢?
2. TextHandler和JSONHandler
當我們想要日志以key-value格式呈現(xiàn)時,我們可以用下面這種方式:
h := slog.NewTextHandler(os.Stderr, nil) l := slog.New(h) l.Info("greeting", "name", "xxx")
最終結果:
time=2023-09-10T21:58:34.144+08:00 level=INFO msg=greeting name=xxx
當我們想要日志以json格式呈現(xiàn)時,我們可以使用下面這種方式:
h1 := slog.NewJSONHandler(os.Stderr, nil) l1 := slog.New(h1) l1.Info("greeting", "name", "xxx")
最終結果:
{"time":"2023-09-10T22:00:04.687003+08:00","level":"INFO","msg":"greeting","name":"xxx"}
slog.NewJSONHandler函數(shù)和slog.NewTextHandler函數(shù)都會返回一個JsonHandler結構體或是TextHandler的引用,這個結構體會被slog.new函數(shù)接受,該函數(shù)返回一個Logger結構體的引用,這個logger結構體包含Handler接口,擁有一系列日志相關函數(shù),是我們最終打印日志的地方
func NewJSONHandler(w io.Writer, opts *HandlerOptions) *JSONHandler {} func NewTextHandler(w io.Writer, opts *HandlerOptions) *TextHandler {} func New(h Handler) *Logger {} type Logger struct { handler Handler // for structured logging } type Handler interface {}
如此,我們實現(xiàn)了基本的日志結構化輸出。
3.日志配置
我們通過對slog.HandlerOptions配置,可以實現(xiàn)例如 是否輸出日志來源 等設置;
s := &slog.HandlerOptions{ AddSource: true, } slog.SetDefault(slog.New(slog.NewTextHandler(os.Stderr, s))) slog.Info("Test", "greeting", "hello, world")
此處筆者將該s設置為default情況,這樣直接使用slog.info就可以用到之前的配置;
最終結果:
time=2023-09-10T22:11:04.432+08:00 level=INFO source="/Users/wurenyu/Library/Mobile Documents/com~apple~CloudDocs/Go_learn/basic/slog/t1.go:13" msg=Test greeting="hello, world"
可以看到,由于AddSource被設置為true,我們的輸出日志中多了source這一信息;
又由于NewTextHandler,所以日志是以鍵值對的形式輸出的。
再來看這一段代碼:
opts := slog.HandlerOptions{ AddSource: true, Level: slog.LevelError, } slog.SetDefault(slog.New(slog.NewJSONHandler(os.Stderr, &opts))) slog.Info("open file for reading", "name", "foo.txt", "path", "/home/tonybai/demo/foo.txt") slog.Error("open file error", "err", os.ErrNotExist, "status", 2)
在slog配置中將Level設置為了LevelError,如此,將只能使用slog.error這一級別;
最終輸出結果:
{"time":"2023-09-10T22:13:44.493714+08:00","level":"ERROR","source":{"function":"main.main","file":"/Users/wurenyu/Library/Mobile Documents/com~apple~CloudDocs/Go_learn/basic/slog/t1.go","line":16},"msg":"open file error","err":"file does not exist","status":2}
4. Group形式輸出日志
baseLogger := slog.New(slog.NewJSONHandler(os.Stdout, nil)) groupedLogger := baseLogger.WithGroup("TTT") // Log with the grouped logger groupedLogger.Info("This log entry includes module information.", "test1", "answer1") groupedLogger.Warn("This log entry also includes module information.", "test2", "answer2")
上述代碼首先生成一個叫做baseLogger的logger,然后在這個logger上調(diào)用方法WithGroup,并傳入?yún)?shù)“TTT”,后面兩行分別輸出info和warn級別的日志;
最終結果如下:
{"time":"2023-09-10T22:23:28.527786+08:00","level":"INFO","msg":"This log entry includes module information.","TTT":{"test1":"answer1"}}
{"time":"2023-09-10T22:23:28.528019+08:00","level":"WARN","msg":"This log entry also includes module information.","TTT":{"test2":"answer2"}}
可以看到,在groupLogger后面加上的鍵值對都被加在了TTT后面;
不過值得關注的是,slog是支持給logger自定義字段的,給一個logger加上一個屬性之后,每次用這個logger輸出日志,都會輸出這個屬性對應的鍵值對,而這個信息不會被包含在WithGroup函數(shù)傳入的參數(shù)后面。
風格
個人認為一般不需要在msg中直接傳入代碼中的數(shù)據(jù),msg中應該盡量直接使用constant常量,這樣更可控。
WHAT
以下是slog大致的架構:
全文終。
以上就是go1.21中slog日志包用法入門的詳細內(nèi)容,更多關于go slog的資料請關注腳本之家其它相關文章!
相關文章
Go語言RPC Authorization進行簡單ip安全驗證的方法
這篇文章主要介紹了Go語言RPC Authorization進行簡單ip安全驗證的方法,實例分析了Go語言進行ip驗證的技巧,需要的朋友可以參考下2015-03-03golang實現(xiàn)多協(xié)程下載文件(支持斷點續(xù)傳)
本文主要介紹了golang實現(xiàn)多協(xié)程下載文件,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-11-11