go日志框架之logrus

本文轉(zhuǎn)載自姜總

golang日志庫(kù)

golang標(biāo)準(zhǔn)庫(kù)的日志框架非常簡(jiǎn)單,僅僅提供了print,panic和fatal三個(gè)函數(shù)對(duì)于更精細(xì)的日志級(jí)別业扒、日志文件分割以及日志分發(fā)等方面并沒有提供支持。所以催生了很多第三方的日志庫(kù),但是在golang的世界里秀睛,沒有一個(gè)日志庫(kù)像slf4j那樣在Java中具有絕對(duì)統(tǒng)治地位。golang中莲祸,流行的日志框架包括logrus蹂安、zap、zerolog锐帜、seelog等田盈。
logrus是目前Github上star數(shù)量最多的日志庫(kù),目前(2018.08缴阎,下同)star數(shù)量為8119允瞧,fork數(shù)為1031。logrus功能強(qiáng)大蛮拔,性能高效述暂,而且具有高度靈活性,提供了自定義插件的功能建炫。很多開源項(xiàng)目畦韭,如docker,prometheus等踱卵,都是用了logrus來記錄其日志廊驼。
zap是Uber推出的一個(gè)快速、結(jié)構(gòu)化的分級(jí)日志庫(kù)惋砂。具有強(qiáng)大的ad-hoc分析功能妒挎,并且具有靈活的儀表盤。zap目前在GitHub上的star數(shù)量約為4.3k西饵。
seelog提供了靈活的異步調(diào)度酝掩、格式化和過濾功能。目前在GitHub上也有約1.1k眷柔。

logrus特性

logrus具有以下特性:

  • 完全兼容golang標(biāo)準(zhǔn)庫(kù)日志模塊:logrus擁有六種日志級(jí)別:debug期虾、info、warn驯嘱、error镶苞、fatal和panic,這是golang標(biāo)準(zhǔn)庫(kù)日志模塊的API的超集鞠评。如果您的項(xiàng)目使用標(biāo)準(zhǔn)庫(kù)日志模塊茂蚓,完全可以以最低的代價(jià)遷移到logrus上。
  • 可擴(kuò)展的Hook機(jī)制:允許使用者通過hook的方式將日志分發(fā)到任意地方,如本地文件系統(tǒng)聋涨、標(biāo)準(zhǔn)輸出晾浴、logstash、elasticsearch或者mq等牍白,或者通過hook定義日志內(nèi)容和格式等脊凰。
  • 可選的日志輸出格式:logrus內(nèi)置了兩種日志格式,JSONFormatterTextFormatter茂腥,如果這兩個(gè)格式不滿足需求狸涌,可以自己動(dòng)手實(shí)現(xiàn)接口Formatter,來定義自己的日志格式础芍。
  • Field機(jī)制:logrus鼓勵(lì)通過Field機(jī)制進(jìn)行精細(xì)化的杈抢、結(jié)構(gòu)化的日志記錄,而不是通過冗長(zhǎng)的消息來記錄日志仑性。
  • logrus是一個(gè)可插拔的、結(jié)構(gòu)化的日志框架右蹦。

logrus的使用

第一個(gè)示例

最簡(jiǎn)單的使用logrus的示例如下:

package main

import (
  log "github.com/sirupsen/logrus"
)

func main() {
  log.WithFields(log.Fields{
    "animal": "walrus",
  }).Info("A walrus appears")
}

上面代碼執(zhí)行后诊杆,標(biāo)準(zhǔn)輸出上輸出如下:

time="2018-08-11T15:42:22+08:00" level=info msg="A walrus appears" animal=walrus

logrus與golang標(biāo)準(zhǔn)庫(kù)日志模塊完全兼容,因此您可以使用log“github.com/sirupsen/logrus”替換所有日志導(dǎo)入何陆。
logrus可以通過簡(jiǎn)單的配置晨汹,來定義輸出、格式或者日志級(jí)別等贷盲。

package main

import (
    "os"
    log "github.com/sirupsen/logrus"
)

func init() {
    // 設(shè)置日志格式為json格式
    log.SetFormatter(&log.JSONFormatter{})

    // 設(shè)置將日志輸出到標(biāo)準(zhǔn)輸出(默認(rèn)的輸出為stderr淘这,標(biāo)準(zhǔn)錯(cuò)誤)
    // 日志消息輸出可以是任意的io.writer類型
    log.SetOutput(os.Stdout)

    // 設(shè)置日志級(jí)別為warn以上
    log.SetLevel(log.WarnLevel)
}

func main() {
    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!")
}

Logger

logger是一種相對(duì)高級(jí)的用法, 對(duì)于一個(gè)大型項(xiàng)目, 往往需要一個(gè)全局的logrus實(shí)例,即logger對(duì)象來記錄項(xiàng)目所有的日志巩剖。如:

package main

import (
    "github.com/sirupsen/logrus"
    "os"
)

// logrus提供了New()函數(shù)來創(chuàng)建一個(gè)logrus的實(shí)例铝穷。
// 項(xiàng)目中,可以創(chuàng)建任意數(shù)量的logrus實(shí)例佳魔。
var log = logrus.New()

func main() {
    // 為當(dāng)前l(fā)ogrus實(shí)例設(shè)置消息的輸出曙聂,同樣地,
    // 可以設(shè)置logrus實(shí)例的輸出到任意io.writer
    log.Out = os.Stdout

    // 為當(dāng)前l(fā)ogrus實(shí)例設(shè)置消息輸出格式為json格式鞠鲜。
    // 同樣地宁脊,也可以單獨(dú)為某個(gè)logrus實(shí)例設(shè)置日志級(jí)別和hook,這里不詳細(xì)敘述贤姆。
    log.Formatter = &logrus.JSONFormatter{}

    log.WithFields(logrus.Fields{
        "animal": "walrus",
        "size":   10,
    }).Info("A group of walrus emerges from the ocean")
}

Fields

前一章提到過榆苞,logrus不推薦使用冗長(zhǎng)的消息來記錄運(yùn)行信息,它推薦使用Fields來進(jìn)行精細(xì)化的霞捡、結(jié)構(gòu)化的信息記錄坐漏。
例如下面的記錄日志的方式:

log.Fatalf("Failed to send event %s to topic %s with key %d", event, topic, key)

在logrus中不太提倡,logrus鼓勵(lì)使用以下方式替代之:

<div class="se-preview-section-delimiter"></div>

log.WithFields(log.Fields{
  "event": event,
  "topic": topic,
  "key": key,
}).Fatal("Failed to send event")

前面的WithFields API可以規(guī)范使用者按照其提倡的方式記錄日志。但是WithFields依然是可選的仙畦,因?yàn)槟承﹫?chǎng)景下输涕,使用者確實(shí)只需要記錄儀一條簡(jiǎn)單的消息。

通常慨畸,在一個(gè)應(yīng)用中莱坎、或者應(yīng)用的一部分中,都有一些固定的Field寸士。比如在處理用戶http請(qǐng)求時(shí)檐什,上下文中,所有的日志都會(huì)有request_iduser_ip弱卡。為了避免每次記錄日志都要使用log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip})乃正,我們可以創(chuàng)建一個(gè)logrus.Entry實(shí)例,為這個(gè)實(shí)例設(shè)置默認(rèn)Fields婶博,在上下文中使用這個(gè)logrus.Entry實(shí)例記錄日志即可瓮具。

requestLogger := log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip})
requestLogger.Info("something happened on that request") # will log request_id and user_ip
requestLogger.Warn("something not great happened")

Hook

logrus最令人心動(dòng)的功能就是其可擴(kuò)展的HOOK機(jī)制了,通過在初始化時(shí)為logrus添加hook凡人,logrus可以實(shí)現(xiàn)各種擴(kuò)展功能名党。

Hook接口

logrus的hook接口定義如下,其原理是每此寫入日志時(shí)攔截挠轴,修改logrus.Entry传睹。

// logrus在記錄Levels()返回的日志級(jí)別的消息時(shí)會(huì)觸發(fā)HOOK,
// 按照Fire方法定義的內(nèi)容修改logrus.Entry岸晦。
type Hook interface {
    Levels() []Level
    Fire(*Entry) error
}

一個(gè)簡(jiǎn)單自定義hook如下欧啤,DefaultFieldHook定義會(huì)在所有級(jí)別的日志消息中加入默認(rèn)字段appName="myAppName"

type DefaultFieldHook struct {
}

func (hook *DefaultFieldHook) Fire(entry *log.Entry) error {
    entry.Data["appName"] = "MyAppName"
    return nil
}

func (hook *DefaultFieldHook) Levels() []log.Level {
    return log.AllLevels
}

hook的使用也很簡(jiǎn)單启上,在初始化前調(diào)用log.AddHook(hook)添加相應(yīng)的hook即可邢隧。

logrus官方僅僅內(nèi)置了syslog的hook
此外碧绞,但Github也有很多第三方的hook可供使用府框,文末將提供一些第三方HOOK的連接。

問題與解決方案

盡管logrus有諸多優(yōu)點(diǎn)讥邻,但是為了靈活性和可擴(kuò)展性迫靖,官方也削減了很多實(shí)用的功能,例如:

  • 沒有提供行號(hào)和文件名的支持
  • 輸出到本地文件系統(tǒng)沒有提供日志分割功能
  • 官方?jīng)]有提供輸出到ELK等日志處理中心的功能

但是這些功能都可以通過自定義hook來實(shí)現(xiàn)兴使。

記錄文件名和行號(hào)

logrus的一個(gè)很致命的問題就是沒有提供文件名和行號(hào)系宜,這在大型項(xiàng)目中通過日志定位問題時(shí)有諸多不便。Github上的logrus的issue#63:Log filename and line number創(chuàng)建于2014年发魄,四年過去了仍是open狀態(tài)~~~
網(wǎng)上給出的解決方案分位兩類盹牧,一就是自己實(shí)現(xiàn)一個(gè)hook俩垃;二就是通過裝飾器包裝logrus.Entry。兩種方案網(wǎng)上都有很多代碼汰寓,但是大多無法正常工作口柳。但總體來說,解決問題的思路都是對(duì)的:通過標(biāo)準(zhǔn)庫(kù)的runtime模塊獲取運(yùn)行時(shí)信息有滑,并從中提取文件名跃闹,行號(hào)和調(diào)用函數(shù)名。

標(biāo)準(zhǔn)庫(kù)runtime模塊的Caller(skip int)函數(shù)可以返回當(dāng)前goroutine調(diào)用棧中的文件名毛好,行號(hào)望艺,函數(shù)信息等,參數(shù)skip表示表示返回的棧幀的層次肌访,0表示runtime.Caller的調(diào)用著找默。返回值包括響應(yīng)棧幀層次的pc(程序計(jì)數(shù)器),文件名和行號(hào)信息吼驶。為了提高效率惩激,我們先通過跟蹤調(diào)用棧發(fā)現(xiàn),從runtime.Caller()的調(diào)用者開始蟹演,到記錄日志的生成代碼之間咧欣,大概有8到11層左右,所有我們?cè)趆ook中循環(huán)第8到11層調(diào)用棧應(yīng)該可以找到日志記錄的生產(chǎn)代碼轨帜。

這里寫圖片描述

此外,runtime.FuncForPC(pc uintptr) *Func可以返回指定pc的函數(shù)信息衩椒。
所有我們要實(shí)現(xiàn)的hook也是基于以上原理蚌父,使用runtime.Caller()依次循環(huán)調(diào)用棧的第7~11層,過濾掉sirupsen包內(nèi)容毛萌,那么第一個(gè)非siupsenr包就認(rèn)為是我們的生產(chǎn)代碼了苟弛,并返回pc以便通過runtime.FuncForPC()獲取函數(shù)名稱。然后將文件名阁将、行號(hào)和函數(shù)名組裝為source字段塞到logrus.Entry中即可膏秫。

time="2018-08-11T19:10:15+08:00" level=warning msg="postgres_exporter is ready for scraping on 0.0.0.0:9295..." source="postgres_exporter/main.go:60:main()"
time="2018-08-11T19:10:17+08:00" level=error msg="!!!msb info not found" source="postgres/postgres_query.go:63:QueryPostgresInfo()"
time="2018-08-11T19:10:17+08:00" level=error msg="get postgres instances info failed, scrape metrics failed, error:msb env not found" source="collector/exporter.go:71:Scrape()"

日志本地文件分割

logrus本身不帶日志本地文件分割功能,但是我們可以通過file-rotatelogs進(jìn)行日志本地文件分割做盅。 每次當(dāng)我們寫入日志的時(shí)候缤削,logrus都會(huì)調(diào)用file-rotatelogs來判斷日志是否要進(jìn)行切分。關(guān)于本地日志文件分割的例子網(wǎng)上很多吹榴,這里不再詳細(xì)介紹亭敢,奉上代碼:

import (
    "github.com/lestrrat-go/file-rotatelogs"
    "github.com/rifflock/lfshook"
    log "github.com/sirupsen/logrus"
    "time"
)

func newLfsHook(logLevel *string, maxRemainCnt uint) log.Hook {
    writer, err := rotatelogs.New(
        logName+".%Y%m%d%H",
        // WithLinkName為最新的日志建立軟連接,以方便隨著找到當(dāng)前日志文件
        rotatelogs.WithLinkName(logName),

        // WithRotationTime設(shè)置日志分割的時(shí)間图筹,這里設(shè)置為一小時(shí)分割一次
        rotatelogs.WithRotationTime(time.Hour),

        // WithMaxAge和WithRotationCount二者只能設(shè)置一個(gè)帅刀,
        // WithMaxAge設(shè)置文件清理前的最長(zhǎng)保存時(shí)間让腹,
        // WithRotationCount設(shè)置文件清理前最多保存的個(gè)數(shù)。
        //rotatelogs.WithMaxAge(time.Hour*24),
        rotatelogs.WithRotationCount(maxRemainCnt),
    )

    if err != nil {
        log.Errorf("config local file system for logger error: %v", err)
    }

    level, ok := logLevels[*logLevel]

    if ok {
        log.SetLevel(level)
    } else {
        log.SetLevel(log.WarnLevel)
    }

    lfsHook := lfshook.NewHook(lfshook.WriterMap{
        log.DebugLevel: writer,
        log.InfoLevel:  writer,
        log.WarnLevel:  writer,
        log.ErrorLevel: writer,
        log.FatalLevel: writer,
        log.PanicLevel: writer,
    }, &log.TextFormatter{DisableColors: true})

    return lfsHook
}

使用上述本地日志文件切割的效果如下:


這里寫圖片描述

將日志發(fā)送到elasticsearch

將日志發(fā)送到elasticsearch是很多日志監(jiān)控系統(tǒng)的選擇扣溺,將logrus日志發(fā)送到elasticsearch的原理是在hook的每次fire調(diào)用時(shí)骇窍,使用golang的es客戶端將日志信息寫到elasticsearch。elasticsearch官方?jīng)]有提供golang客戶端锥余,但是有很多第三方的go語言客戶端可供使用腹纳,我們選擇elastic。elastic提供了豐富的文檔哈恰,以及Java中的流式接口只估,使用起來非常方便。

client, err := elastic.NewClient(elastic.SetURL("http://localhost:9200"))
    if err != nil {
        log.Panic(err)
    }

// Index a tweet (using JSON serialization)
tweet1 := Tweet{User: "olivere", Message: "Take Five", Retweets: 0}
put1, err := client.Index().
    Index("twitter").
    Type("tweet").
    Id("1").
    BodyJson(tweet1).
    Do(context.Background())

考慮到logrus的Fields機(jī)制着绷,可以實(shí)現(xiàn)如下數(shù)據(jù)格式:

msg := struct {
    Host      string
    Timestamp string `json:"@timestamp"`
    Message   string
    Data      logrus.Fields
    Level     string
}

其中Host記錄產(chǎn)生日志主機(jī)信息蛔钙,在創(chuàng)建hook是指定。其他數(shù)據(jù)需要從logrus.Entry中取得荠医。測(cè)試過程我們選擇按照此原理實(shí)現(xiàn)的第三方HOOK:elogrus吁脱。其使用如下:

import (
    "github.com/olivere/elastic"
    "gopkg.in/sohlich/elogrus"
)

func initLog() {
    client, err := elastic.NewClient(elastic.SetURL("http://localhost:9200"))
    if err != nil {
        log.Panic(err)
    }
    hook, err := elogrus.NewElasticHook(client, "localhost", log.DebugLevel, "mylog")
    if err != nil {
        log.Panic(err)
    }
    log.AddHook(hook)
}

從Elasticsearch查詢得到日志存儲(chǔ),效果如下:

GET http://localhost:9200/mylog/_search

HTTP/1.1 200 OK
content-type: application/json; charset=UTF-8
transfer-encoding: chunked

{
  "took": 1,
  "timed_out": false,
  "_shards": {
    "total": 5,
    "successful": 5,
    "failed": 0
  },
  "hits": {
    "total": 2474,
    "max_score": 1.0,
    "hits": [
      {
        "_index": "mylog",
        "_type": "log",
        "_id": "AWUw13jWnMZReb-jHQup",
        "_score": 1.0,
        "_source": {
          "Host": "localhost",
          "@timestamp": "2018-08-13T01:12:32.212818666Z",
          "Message": "!!!msb info not found",
          "Data": {},
          "Level": "ERROR"
        }
      },
      {
        "_index": "mylog",
        "_type": "log",
        "_id": "AWUw13jgnMZReb-jHQuq",
        "_score": 1.0,
        "_source": {
          "Host": "localhost",
          "@timestamp": "2018-08-13T01:12:32.223103348Z",
          "Message": "get postgres instances info failed, scrape metrics failed, error:msb env not found",
          "Data": {
            "source": "collector/exporter.go:71:Scrape()"
          },
          "Level": "ERROR"
        }
      },
      //...
      {
        "_index": "mylog",
        "_type": "log",
        "_id": "AWUw2f1enMZReb-jHQu_",
        "_score": 1.0,
        "_source": {
          "Host": "localhost",
          "@timestamp": "2018-08-13T01:15:17.212546892Z",
          "Message": "!!!msb info not found",
          "Data": {
            "source": "collector/exporter.go:71:Scrape()"
          },
          "Level": "ERROR"
        }
      },
      {
        "_index": "mylog",
        "_type": "log",
        "_id": "AWUw2NhmnMZReb-jHQu1",
        "_score": 1.0,
        "_source": {
          "Host": "localhost",
          "@timestamp": "2018-08-13T01:14:02.21276903Z",
          "Message": "!!!msb info not found",
          "Data": {},
          "Level": "ERROR"
        }
      }
    ]
  }
}

Response code: 200 (OK); Time: 16ms; Content length: 3039 bytes

將日志發(fā)送到其他位置

將日志發(fā)送到日志中心也是logrus所提倡的彬向,雖然沒有提供官方支持兼贡,但是目前Github上有很多第三方hook可供使用:

等等凿蒜,上述第三方hook我這里沒有具體驗(yàn)證,大家可以根據(jù)需要自行嘗試胁黑。

其他注意事項(xiàng)

Fatal處理

和很多日志框架一樣废封,logrus的Fatal系列函數(shù)會(huì)執(zhí)行os.Exit(1)。但是logrus提供可以注冊(cè)一個(gè)或多個(gè)fatal handler函數(shù)的接口logrus.RegisterExitHandler(handler func() {} )丧蘸,讓logrus在執(zhí)行os.Exit(1)之前進(jìn)行相應(yīng)的處理漂洋。fatal handler可以在系統(tǒng)異常時(shí)調(diào)用一些資源釋放api等,讓應(yīng)用正確的關(guān)閉力喷。

線程安全

默認(rèn)情況下刽漂,logrus的api都是線程安全的,其內(nèi)部通過互斥鎖來保護(hù)并發(fā)寫冗懦∷幔互斥鎖工作于調(diào)用hooks或者寫日志的時(shí)候,如果不需要鎖披蕉,可以調(diào)用logger.SetNoLock()來關(guān)閉之颈畸∥谄妫可以關(guān)閉logrus互斥鎖的情形包括:

  • 沒有設(shè)置hook,或者所有的hook都是線程安全的實(shí)現(xiàn)眯娱。
  • 寫日志到logger.Out已經(jīng)是線程安全的了礁苗,如logger.Out已經(jīng)被鎖保護(hù),或者寫文件時(shí)徙缴,文件是以O_APPEND方式打開的试伙,并且每次寫操作都小于4k。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末于样,一起剝皮案震驚了整個(gè)濱河市疏叨,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌穿剖,老刑警劉巖蚤蔓,帶你破解...
    沈念sama閱讀 221,406評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異糊余,居然都是意外死亡秀又,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,395評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門贬芥,熙熙樓的掌柜王于貴愁眉苦臉地迎上來吐辙,“玉大人,你說我怎么就攤上這事蘸劈』杷眨” “怎么了?”我有些...
    開封第一講書人閱讀 167,815評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵威沫,是天一觀的道長(zhǎng)捷雕。 經(jīng)常有香客問我,道長(zhǎng)壹甥,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,537評(píng)論 1 296
  • 正文 為了忘掉前任壶熏,我火速辦了婚禮句柠,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘棒假。我一直安慰自己溯职,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,536評(píng)論 6 397
  • 文/花漫 我一把揭開白布帽哑。 她就那樣靜靜地躺著谜酒,像睡著了一般。 火紅的嫁衣襯著肌膚如雪妻枕。 梳的紋絲不亂的頭發(fā)上僻族,一...
    開封第一講書人閱讀 52,184評(píng)論 1 308
  • 那天粘驰,我揣著相機(jī)與錄音,去河邊找鬼述么。 笑死蝌数,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的度秘。 我是一名探鬼主播顶伞,決...
    沈念sama閱讀 40,776評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼剑梳!你這毒婦竟也來了唆貌?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,668評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤垢乙,失蹤者是張志新(化名)和其女友劉穎锨咙,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體侨赡,經(jīng)...
    沈念sama閱讀 46,212評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蓖租,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,299評(píng)論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了羊壹。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蓖宦。...
    茶點(diǎn)故事閱讀 40,438評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖油猫,靈堂內(nèi)的尸體忽然破棺而出稠茂,到底是詐尸還是另有隱情,我是刑警寧澤情妖,帶...
    沈念sama閱讀 36,128評(píng)論 5 349
  • 正文 年R本政府宣布睬关,位于F島的核電站,受9級(jí)特大地震影響毡证,放射性物質(zhì)發(fā)生泄漏电爹。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,807評(píng)論 3 333
  • 文/蒙蒙 一料睛、第九天 我趴在偏房一處隱蔽的房頂上張望丐箩。 院中可真熱鬧,春花似錦恤煞、人聲如沸屎勘。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,279評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽概漱。三九已至,卻和暖如春喜喂,著一層夾襖步出監(jiān)牢的瞬間瓤摧,已是汗流浹背竿裂。 一陣腳步聲響...
    開封第一講書人閱讀 33,395評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留姻灶,地道東北人铛绰。 一個(gè)月前我還...
    沈念sama閱讀 48,827評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像产喉,于是被迫代替她去往敵國(guó)和親捂掰。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,446評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容

  • (http://www.cnblogs.com/zhangchenliang/p/4546352.html) 1曾沈、...
    凌雲(yún)木閱讀 2,428評(píng)論 0 2
  • 在我們開發(fā)程序后这嚣,如果有一些問題需要對(duì)程序進(jìn)行調(diào)試的時(shí)候,日志是必不可少的塞俱,這是我們分析程序問題常用的手段姐帚。 日志...
    豆瓣奶茶閱讀 18,365評(píng)論 0 22
  • 在應(yīng)用程序中添加日志記錄總的來說基于三個(gè)目的:監(jiān)視代碼中變量的變化情況,周期性的記錄到文件中供其他應(yīng)用進(jìn)行統(tǒng)計(jì)分析...
    時(shí)待吾閱讀 4,991評(píng)論 0 6
  • 爸爸年輕的時(shí)候是個(gè)貨車司機(jī)障涯。去過很多地方罐旗,也很少管我。那些年父母都很忙唯蝶,忙于養(yǎng)家糊口九秀。一轉(zhuǎn)眼到了我忙于奔命的...
    蘇肉閱讀 808評(píng)論 3 0
  • 當(dāng)一個(gè)人游走在死神的邊緣,他會(huì)想些什么呢粘我? 2004年鼓蜒,楊絳先生生病入院,在病床上的時(shí)候征字,她就一直在思索一個(gè)題目都弹,...
    林子說心理閱讀 881評(píng)論 0 0