Golang學(xué)習(xí)筆記之日志log嫌变、zap

一吨艇、log日志包

log支持并發(fā)操作。其結(jié)構(gòu)定義如下:

type Logger struct {
    mu sync.Mutex     // ensures atomic writes; protects the following fields
    prefix string     // prefix to write at beginning of each line // ?志?前綴
    flag int          // properties // ?志打印格式標(biāo)志腾啥,?于指定每??志的打印格式
    out io.Writer     // destination for output // ?于指定?志輸出位置东涡,理論上可以是任務(wù)地?,只要實(shí)現(xiàn)了io.Writer接?就?
    buf []byte        // for accumulating text to write // ?志內(nèi)容
}

log基本日志格式

Ldate       = 1 << iota     //  形如  2009/01/23  的?期
Ltime                       //  形如  01:23:23            的時間
Lmicroseconds               //  形如  01:23:23.123123         的時間
Llongfile                   //  全路徑?件名和?號:  /a/b/c/d.go:23  
Lshortfile                  //  ?件名和?號: d.go:23 
LstdFlags   = Ldate | Ltime //  ?期和時間

(1)Golang's log模塊主要提供了3類接口碑宴。分別是 “Print 软啼、Panic 、Fatal ”延柠,對每一類接口其提供了3中調(diào)用方式祸挪,分別是 "Xxxx 、Xxxxln 贞间、Xxxxf"贿条,基本和fmt中的相關(guān)函數(shù)類似。

? log.Print:打印日志增热,和fmt.包沒什么區(qū)別整以,只是加上了上面的日志格式
? log.Fatal :,會先將日志內(nèi)容打印到標(biāo)準(zhǔn)輸出峻仇,接著調(diào)用系統(tǒng)的os.exit(1) 接口公黑,退出程序并返回狀態(tài) 1 。但是有一點(diǎn)需要注意摄咆,由于是直接調(diào)用系統(tǒng)接口退出凡蚜,defer函數(shù)不會被調(diào)用。
? log.Panic:該函數(shù)把日志內(nèi)容刷到標(biāo)準(zhǔn)錯誤后調(diào)用 panic 函數(shù)吭从,
demo

package main

import (
    "fmt"
    "log"
)
//fatal
func testDeferfatal() {
    defer func() {
        fmt.Println("--first--")
    }()
    log.Fatalln("test for defer Fatal")
}
//panic
func testDeferpanic() {
    defer func() {
        fmt.Println("--first--")
        if err := recover(); err != nil {
            fmt.Println(err)
        }
    }()
    log.Panicln("test for defer Panic")
    defer func() {
        fmt.Println("--second--")
    }()
}
func main() {
    arr := []int{2, 3}

    log.Print("Print array ", arr, "\n")
    log.Println("Println array", arr)
    log.Printf("Printf array with item [%d,%d]\n", arr[0], arr[1])

    testDeferpanic()


    testDeferfatal()
}

輸出為
2018/12/17 21:28:33 Print array [2 3]
2018/12/17 21:28:33 Println array [2 3]
2018/12/17 21:28:33 Printf array with item [2,3]
2018/12/17 21:28:33 test for defer Panic
--first--
test for defer Panic

2018/12/17 21:28:33 test for defer Fatal
exit status 1

(2)你也可以自定義Logger類型

log.Logger提供了一個New方法用來創(chuàng)建對象
函數(shù)原型
func New(out io.Writer, prefix string, flag int) *Logger
①輸出位置out朝蜘,是一個io.Writer對象,該對象可以是一個文件也可以是實(shí)現(xiàn)了該接口的對象涩金。通常我們可以用這個來指定日志輸出到哪個文件
②prefix 我們在前面已經(jīng)看到谱醇,就是在日志內(nèi)容前面的東西。我們可以將其置為 "[Info]" 步做、 "[Warning]"等來幫助區(qū)分日志級別副渴。
③flags 是一個選項(xiàng),顯示日志開頭的東西全度,可選的值見前面所述
demo

func main() {
    fileName := "/Users/zt/Desktop/Info_First.log"http://路徑+文件名
    logFile, err := os.Create(fileName)
    defer logFile.Close()
    if err != nil {
        log.Fatalln("open file error")
    }
    debugLog := log.New(logFile, "[Info]", log.Llongfile)
    debugLog.Println("A Info message here")
    debugLog.SetPrefix("[Debug]")
    debugLog.Println("A Debug Message here ")
}
二煮剧、Zap日志包使用

uber開源的高性能日志庫
go get go.uber.org/zap
demo

func panic() {
    if err := recover(); err != nil {
        fmt.Println(err)
    }
}
func main() {
    url := "Hello"
    logger, _ := zap.NewProduction()
    //logger, _ := zap.NewDevelopment()

    defer panic()
    //Sync刷新任何緩沖的日志條目。
    defer logger.Sync()
    logger.Info("failed to fetch URL",
        // Structured context as strongly typed Field values.
        zap.String("url", url),
        zap.Int("attempt", 3),
        zap.Duration("backoff", time.Second),
    )
    logger.Warn("debug log", zap.String("level", url))
    logger.Error("Error Message", zap.String("error", url))
    logger.Panic("Panic log", zap.String("level", url))
}

(1)通過HTTP接口動態(tài)的改變?nèi)罩炯墑e

demo

func main() {
    alevel := zap.NewAtomicLevel()
    http.HandleFunc("/handle/level", alevel.ServeHTTP)
    go func() {
        if err := http.ListenAndServe(":9090", nil); err != nil {
            panic(err)
        }
    }()
    // 默認(rèn)是Info級別
    logcfg := zap.NewProductionConfig()
    logcfg.Level = alevel
    logger, err := logcfg.Build()
    if err != nil {
        fmt.Println("err", err)
    }
    defer logger.Sync()
    for i := 0; i < 1000; i++ {
        time.Sleep(1 * time.Second)
        logger.Debug("debug log", zap.String("level", alevel.String()))
        logger.Info("Info log", zap.String("level", alevel.String()))
    }
}

查看日志級別
curl http://localhost:9090/handle/level
輸出


調(diào)整日志級別(可選值 “debug” “info” “warn” “error” 等)
curl -XPUT --data '{"level":"debug"}' http://localhost:9090/handle/level
輸出

當(dāng)然也可以使用之前在Gin說過的工具RESTClient來模擬讼载,

(2)將日志進(jìn)行序列化文件:lumberjack

支持文件按大小或者時間歸檔
GitHub地址:https://github.com/natefinch/lumberjack

go get github.com/natefinch/lumberjack

package main

import (
    "go.uber.org/zap"
    "go.uber.org/zap/zapcore"
    lumberjack "gopkg.in/natefinch/lumberjack.v2"
)
func initLogger(logpath string, loglevel string) *zap.Logger {
    hook := lumberjack.Logger{
        Filename:   logpath, // ?志?件路徑
        MaxSize:    1024,    // megabytes
        MaxBackups: 3,       // 最多保留3個備份
        MaxAge:     7,       //days
        Compress:   true,    // 是否壓縮 disabled by default
    }
    w := zapcore.AddSync(&hook)
    var level zapcore.Level
    switch loglevel {
    case "debug":
        level = zap.DebugLevel
    case "info":
        level = zap.InfoLevel
    case "error":
        level = zap.ErrorLevel
    default:
        level = zap.InfoLevel
    }
    encoderConfig := zap.NewProductionEncoderConfig()
    encoderConfig.EncodeTime = zapcore.ISO8601TimeEncoder
    core := zapcore.NewCore(
        zapcore.NewConsoleEncoder(encoderConfig),
        w,
        level,
    )
    logger := zap.New(core)
    logger.Info("DefaultLogger init success")
    return logger
}
func main() {
    logger := initLogger("all.log", "info")
    logger.Info("test log", zap.Int("line", 47))
    logger.Warn("testlog", zap.Int("line", 47))
}

在當(dāng)前文件夾下的all.log


最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末轿秧,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子咨堤,更是在濱河造成了極大的恐慌菇篡,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,544評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件一喘,死亡現(xiàn)場離奇詭異驱还,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)凸克,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評論 3 392
  • 文/潘曉璐 我一進(jìn)店門议蟆,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人萎战,你說我怎么就攤上這事咐容。” “怎么了蚂维?”我有些...
    開封第一講書人閱讀 162,764評論 0 353
  • 文/不壞的土叔 我叫張陵戳粒,是天一觀的道長。 經(jīng)常有香客問我虫啥,道長蔚约,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,193評論 1 292
  • 正文 為了忘掉前任涂籽,我火速辦了婚禮苹祟,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘评雌。我一直安慰自己树枫,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,216評論 6 388
  • 文/花漫 我一把揭開白布柳骄。 她就那樣靜靜地躺著团赏,像睡著了一般。 火紅的嫁衣襯著肌膚如雪耐薯。 梳的紋絲不亂的頭發(fā)上舔清,一...
    開封第一講書人閱讀 51,182評論 1 299
  • 那天,我揣著相機(jī)與錄音曲初,去河邊找鬼体谒。 笑死,一個胖子當(dāng)著我的面吹牛臼婆,可吹牛的內(nèi)容都是我干的抒痒。 我是一名探鬼主播,決...
    沈念sama閱讀 40,063評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼颁褂,長吁一口氣:“原來是場噩夢啊……” “哼故响!你這毒婦竟也來了傀广?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,917評論 0 274
  • 序言:老撾萬榮一對情侶失蹤彩届,失蹤者是張志新(化名)和其女友劉穎伪冰,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體樟蠕,經(jīng)...
    沈念sama閱讀 45,329評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡贮聂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,543評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了寨辩。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片吓懈。...
    茶點(diǎn)故事閱讀 39,722評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖靡狞,靈堂內(nèi)的尸體忽然破棺而出耻警,到底是詐尸還是另有隱情,我是刑警寧澤耍攘,帶...
    沈念sama閱讀 35,425評論 5 343
  • 正文 年R本政府宣布榕栏,位于F島的核電站,受9級特大地震影響蕾各,放射性物質(zhì)發(fā)生泄漏扒磁。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,019評論 3 326
  • 文/蒙蒙 一式曲、第九天 我趴在偏房一處隱蔽的房頂上張望妨托。 院中可真熱鬧,春花似錦吝羞、人聲如沸兰伤。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,671評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽敦腔。三九已至,卻和暖如春恨溜,著一層夾襖步出監(jiān)牢的瞬間符衔,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,825評論 1 269
  • 我被黑心中介騙來泰國打工糟袁, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留判族,地道東北人。 一個月前我還...
    沈念sama閱讀 47,729評論 2 368
  • 正文 我出身青樓项戴,卻偏偏與公主長得像形帮,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,614評論 2 353