您好,登錄后才能下訂單哦!
本文主要介紹 Golang 中最佳日志解決方案,包括常用日志包logrus 的基本使用,如何結合file-rotatelogs 包實現日志文件的輪轉切割兩大話題。
Golang 關于日志處理有很多包可以使用,標準庫提供的 log 包功能比較少,不支持日志級別的精確控制,自定義添加日志字段等。在眾多的日志包中,更推薦使用第三方的 logrus 包,完全兼容自帶的 log 包。logrus 是目前 Github 上 star 數量最多的日志庫,logrus 功能強大,性能高效,而且具有高度靈活性,提供了自定義插件的功能。
很多開源項目,如 docker,prometheus,dejavuzhou/ginbro 等,都是用了 logrus 來記錄其日志。
logrus 特性
logrus 基本使用
package main import ( "os" "github.com/sirupsen/logrus" log "github.com/sirupsen/logrus" ) var logger *logrus.Entry func init() { // 設置日志格式為json格式 log.SetFormatter(&log.JSONFormatter{}) log.SetOutput(os.Stdout) log.SetLevel(log.InfoLevel) logger = log.WithFields(log.Fields{"request_id": "123444", "user_ip": "127.0.0.1"}) } func main() { logger.Info("hello, logrus....") logger.Info("hello, logrus1....") // log.WithFields(log.Fields{ // "animal": "walrus", // "size": 10, // }).Info("A group of walrus emerges from the ocean") // log.WithFields(log.Fields{ // "omg": true, // "number": 122, // }).Warn("The group's number increased tremendously!") // log.WithFields(log.Fields{ // "omg": true, // "number": 100, // }).Fatal("The ice breaks!") }
基于 logrus 和 file-rotatelogs 包實現日志切割
很多時候應用會將日志輸出到文件系統,對于訪問量大的應用來說日志的自動輪轉切割管理是個很重要的問題,如果應用不能妥善處理日志管理,那么會帶來很多不必要的維護開銷:外部工具切割日志、人工清理日志等手段確保不會將磁盤打滿。
file-rotatelogs: When you integrate this to to you app, it automatically write to logs that are rotated from within the app: No more disk-full alerts because you forgot to setup logrotate!
logrus 本身不支持日志輪轉切割功能,需要配合 file-rotatelogs 包來實現,防止日志打滿磁盤。file-rotatelogs 實現了 io.Writer 接口,并且提供了文件的切割功能,其實例可以作為 logrus 的目標輸出,兩者能無縫集成,這也是 file-rotatelogs 的設計初衷:
It's normally expected that this library is used with some other logging service, such as the built-in log library, or loggers such as github.com/lestrrat-go/apache-logformat.
示例代碼:
應用日志文件 /Users/opensource/test/go.log,每隔 1 分鐘輪轉一個新文件,保留最近 3 分鐘的日志文件,多余的自動清理掉。
package main import ( "time" rotatelogs "github.com/lestrrat-go/file-rotatelogs" log "github.com/sirupsen/logrus" ) func init() { path := "/Users/opensource/test/go.log" /* 日志輪轉相關函數 `WithLinkName` 為最新的日志建立軟連接 `WithRotationTime` 設置日志分割的時間,隔多久分割一次 WithMaxAge 和 WithRotationCount二者只能設置一個 `WithMaxAge` 設置文件清理前的最長保存時間 `WithRotationCount` 設置文件清理前最多保存的個數 */ // 下面配置日志每隔 1 分鐘輪轉一個新文件,保留最近 3 分鐘的日志文件,多余的自動清理掉。 writer, _ := rotatelogs.New( path+".%Y%m%d%H%M", rotatelogs.WithLinkName(path), rotatelogs.WithMaxAge(time.Duration(180)*time.Second), rotatelogs.WithRotationTime(time.Duration(60)*time.Second), ) log.SetOutput(writer) //log.SetFormatter(&log.JSONFormatter{}) } func main() { for { log.Info("hello, world!") time.Sleep(time.Duration(2) * time.Second) } }
Golang 標準日志庫 log 使用
雖然 Golang 標準日志庫功能少,但是可以選擇性的了解下,下面為基本使用的代碼示例,比較簡單:
package main import ( "fmt" "log" ) func init() { log.SetPrefix("【UserCenter】") // 設置每行日志的前綴 log.SetFlags(log.LstdFlags | log.Lshortfile | log.LUTC) // 設置日志的抬頭字段 } func main() { log.Println("log...") log.Fatalln("Fatal Error...") fmt.Println("Not print!") }
自定義日志輸出
package main import ( "io" "log" "os" ) var ( Info *log.Logger Warning *log.Logger Error *log.Logger ) func init() { errFile, err := os.OpenFile("errors.log", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0666) if err != nil { log.Fatalln("打開日志文件失敗:", err) } Info = log.New(os.Stdout, "Info:", log.Ldate|log.Ltime|log.Lshortfile) Warning = log.New(os.Stdout, "Warning:", log.Ldate|log.Ltime|log.Lshortfile) Error = log.New(io.MultiWriter(os.Stderr, errFile), "Error:", log.Ldate|log.Ltime|log.Lshortfile) } func main() { Info.Println("Info log...") Warning.Printf("Warning log...") Error.Println("Error log...") }
相關文檔
https://mojotv.cn/2018/12/27/golang-logrus-tutorial
https://github.com/lestrrat-go/file-rotatelogs
https://www.flysnow.org/2017/05/06/go-in-action-go-log.html
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。