欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

golang log4go的日志輸出優(yōu)化詳解

 更新時(shí)間:2017年12月28日 11:08:31   作者:ccpaging  
log4go源于google的一項(xiàng)log工程,但官方已經(jīng)停止維護(hù)更新,下面這篇文章主要給大家介紹了關(guān)于golang log4go的日志輸出優(yōu)化的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來(lái)一起看看吧。

前言

在go語(yǔ)言中,自身已經(jīng)集成了一定log模塊,開(kāi)發(fā)者可以使用go語(yǔ)言自身的log包(import “l(fā)og”) 。也有不少對(duì)自身log的開(kāi)源封裝。對(duì)于一些簡(jiǎn)單的開(kāi)發(fā),自身的log模塊就已經(jīng)足夠應(yīng)付。但是對(duì)一些大型,復(fù)雜的開(kāi)發(fā),log需要分門別類的輸出,或者通過(guò)網(wǎng)絡(luò)進(jìn)行輸出,自身log模塊將難以應(yīng)對(duì)。

當(dāng)前也有一些比較重量級(jí)的log模塊,比如logrus,可以實(shí)現(xiàn)比較復(fù)雜的功能。這里介紹一個(gè)輕量級(jí)的log模塊——log4go

最近又看了一些golang的日志包和相關(guān)的文章,仔細(xì)閱讀了go 1.9.2系統(tǒng)提供的log和go-log,產(chǎn)生了對(duì)log4go的日志輸出進(jìn)行優(yōu)化的想法。

結(jié)構(gòu)化與multiwriter

log使用multiwriter支持多個(gè)日志輸出,用 Mutex 加鎖解決多線程日志輸出的沖突。log4go 則采用結(jié)構(gòu)化編程用 channel 傳遞 LogRecord 日志記錄。

原來(lái)以為 channel 的效率比較高……其實(shí)這是一個(gè)偽命題。channel 是一個(gè)全局加鎖的隊(duì)列,可以用來(lái)加鎖,但效率比較低。因?yàn)樗嗔藗鬟f數(shù)據(jù)、協(xié)調(diào)順序處理、timout等功能,并不僅僅是加鎖。跟Mutex不是一回事兒。

log4go 將屏幕日志輸出 termlog 放在了結(jié)構(gòu)里,這帶來(lái)一個(gè)小問(wèn)題。當(dāng)我們用log4go調(diào)試小程序時(shí),運(yùn)行的太快,termlog 的 goroutine 還沒(méi)有運(yùn)行起來(lái),程序就退出了。結(jié)果屏幕上沒(méi)有顯示日志。這個(gè)問(wèn)題只能通過(guò)在 Close() 時(shí)加延時(shí),等待 goroutine 啟動(dòng)來(lái)解決。然后還要檢查 channel ……

func (f *Filter) Close() {
 if f.closed {
  return
 }
 // sleep at most one second and let go routine running
 // drain the log channel before closing
 for i := 10; i > 0; i-- {
  // Must call Sleep here, otherwise, may panic send on closed channel
  time.Sleep(100 * time.Millisecond)
  if len(f.rec) <= 0 {
   break
  }
 }
 // block write channel
 f.closed = true
 defer f.LogWriter.Close()
 close(f.rec)
 if len(f.rec) <= 0 {
  return
 }
 // drain the log channel and write driect
 for rec := range f.rec {
  f.LogWrite(rec)
 }
}

log直接將格式化日志信息輸出到屏幕,簡(jiǎn)單多了。

試著兼顧兩者,在 log4go 中增加了 writer,直接輸出到屏幕。擬將FileLog,SocketLog作為backend,仍然放在結(jié)構(gòu)里。這樣,調(diào)試小程序和生產(chǎn)程序可以使用同一個(gè)日志庫(kù)。實(shí)測(cè)效率略有降低。不知道 windows 下的 ColorLog 如何,以后再說(shuō)。

在log4go中可以通過(guò)調(diào)用 SetOutput(nil) ,使out = nil 來(lái)關(guān)閉屏幕輸出。

Determine caller func - it's expensive

這句話注釋在 log 源文件中,log4go也要調(diào)用runtime.Caller(skip int)函數(shù)來(lái)獲取源文件名和行號(hào)。它是昂貴的——消耗了CPU。建議在生產(chǎn)環(huán)境中關(guān)閉,log.SetSkip(-1) 。如果要對(duì)log4go進(jìn)行封裝,設(shè)置 log.SetSkip(log.GetSkip()+1) 。

format優(yōu)化

其實(shí),這才是文章的主題。

日志輸出避免不了打印日期和時(shí)間,linux 環(huán)境下還要打印微秒,說(shuō)不定還要打印時(shí)區(qū)。log4go的pattlog.go就是完成這些工作的。

  • 有一個(gè)1秒更新一次的cache機(jī)制。很漂亮。
  • 大量使用字符串格式化函數(shù)——fmt.Sprintf。
  • 返回字符串。而writer一般支持的是[]byte。多做一次轉(zhuǎn)換。
  • 每次都bytes.Splite講format字符串以%字符分解成[][]byte。

在log里邊自備了一個(gè)cheap的itoa函數(shù)。

// Cheap integer to fixed-width decimal ASCII. Give a negative width to avoid zero-padding.
func itoa(buf *[]byte, i int, wid int) {
 // Assemble decimal in reverse order.
 var b [20]byte
 bp := len(b) - 1
 for i >= 10 || wid > 1 {
  wid--
  q := i / 10
  b[bp] = byte('0' + i - q*10)
  bp--
  i = q
 }
 // i < 10
 b[bp] = byte('0' + i)
 *buf = append(*buf, b[bp:]...)
}

用這個(gè)函數(shù)替換日期和時(shí)間的字符串格式化函數(shù)。用[]byte代替string。

優(yōu)化前,log4go 的 benchmark。

BenchmarkFormatLogRecord-4    300000    4480 ns/op
BenchmarkConsoleLog-4     1000000    1748 ns/op
BenchmarkConsoleNotLogged-4    20000000    97.5 ns/op
BenchmarkConsoleUtilLog-4     300000    3496 ns/op
BenchmarkConsoleUtilNotLog-4   20000000    104 ns/op

優(yōu)化后:

BenchmarkFormatLogRecord-4  1000000    1443 ns/op
BenchmarkConsoleLog-4   2000000    982 ns/op
BenchmarkConsoleUtilLog-4   500000    3242 ns/op
BenchmarkConsoleUtilNotLog-4 30000000    48.4 ns/op

格式化日期時(shí)間所花的時(shí)間是原來(lái)的1/3。

打印無(wú)格式化信息所花的時(shí)間是原來(lái)的1/2。

BenchmarkConsoleUtilLog調(diào)用了runtime.Caller,格式化信息,且新增了輸出信息到屏幕的時(shí)間。

字符串格式化——比較昂貴。

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

  • Golang監(jiān)聽(tīng)日志文件并發(fā)送到kafka中

    Golang監(jiān)聽(tīng)日志文件并發(fā)送到kafka中

    這篇文章主要介紹了Golang監(jiān)聽(tīng)日志文件并發(fā)送到kafka中,日志收集項(xiàng)目的準(zhǔn)備中,本文主要講的是利用golang的tail庫(kù),監(jiān)聽(tīng)日志文件的變動(dòng),將日志信息發(fā)送到kafka中?,需要的朋友可以參考一下
    2022-04-04
  • go語(yǔ)言環(huán)境變量設(shè)置全過(guò)程

    go語(yǔ)言環(huán)境變量設(shè)置全過(guò)程

    這篇文章主要介紹了go語(yǔ)言環(huán)境變量設(shè)置全過(guò)程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • 使用Go語(yǔ)言實(shí)現(xiàn)配置文件熱加載功能

    使用Go語(yǔ)言實(shí)現(xiàn)配置文件熱加載功能

    這篇文章主要介紹了使用Go語(yǔ)言實(shí)現(xiàn)配置文件熱加載功能,以及配置文件熱加載包的實(shí)現(xiàn)思路,需要的朋友可以參考下
    2018-03-03
  • Golang自定義結(jié)構(gòu)體轉(zhuǎn)map的操作

    Golang自定義結(jié)構(gòu)體轉(zhuǎn)map的操作

    這篇文章主要介紹了Golang自定義結(jié)構(gòu)體轉(zhuǎn)map的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-12-12
  • 一文詳解Golang的函數(shù)特性

    一文詳解Golang的函數(shù)特性

    函數(shù)是?Golang?中非常重要的組成部分之一,它們提供了代碼的可重用性和組織性。在本文中,我們將深入了解?Golang?函數(shù)的多個(gè)方面,希望對(duì)大家有所幫助
    2023-04-04
  • 一文了解Go 并發(fā)與并行

    一文了解Go 并發(fā)與并行

    并發(fā)性和并行性是是兩個(gè)既有聯(lián)系又有所區(qū)別的概念,本文主要介紹了Go并發(fā)與并行,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-05-05
  • Go 驗(yàn)證字符串中是否包含中文(推薦)

    Go 驗(yàn)證字符串中是否包含中文(推薦)

    這篇文章主要介紹了Go 驗(yàn)證字符串中是否包含中文,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-01-01
  • 帶你在Go?test中體驗(yàn)jest的安裝使用

    帶你在Go?test中體驗(yàn)jest的安裝使用

    這篇文章帶你在Go?test中體驗(yàn)jest的安裝使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • Go快速開(kāi)發(fā)一個(gè)RESTful API服務(wù)

    Go快速開(kāi)發(fā)一個(gè)RESTful API服務(wù)

    這篇文章主要為大家介紹了Go快速開(kāi)發(fā)一個(gè)RESTful API服務(wù),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-06-06
  • 基于Golang設(shè)計(jì)一套可控的定時(shí)任務(wù)系統(tǒng)

    基于Golang設(shè)計(jì)一套可控的定時(shí)任務(wù)系統(tǒng)

    這篇文章主要為大家學(xué)習(xí)介紹了如何基于Golang設(shè)計(jì)一套可控的定時(shí)任務(wù)系統(tǒng),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-07-07

最新評(píng)論