当前头条:如何使用GO语言从零实现日志包
2023-04-18 10:21:54    今日头条
背景

当前的开源日志包有很多,像go中的标准库log包、glog、logrus、zap。它们每种日志包都有相应的应用场景。四种日志包相关对比如下所示:

标准库log

功能简单,不支持日志级别、日志格式。但是使用简单,易于快速上手。大型项目较少使用


【资料图】

glog

提供了日志包的基本功能,像日志级别、格式等。适合一些小项目

logrus

功能强大,不仅实现了基本日志功能,还提供了很多高级功能。适合大型项目

zap

功能强大,性能高,适合对日志性能要求高的项目。另外zap的子包zapcore提供了很多底层日志接口,适合二次开发

从头开发一个日志包,可以让我们了解日志包的底层逻辑,使得我们对日志包有定制需求的时候,可以能够基于开源的日志包实现我们的功能。所以本文以cuslog为例,看看如何实现我们自己的日志包(代码:https://github.com/marmotedu/gopractise-demo/tree/master/log/cuslog)。

代码结构

cuslog目录代码结构

cuslog代码结构

要实现一个日志包,就需要实现下面三个基本的对象,Entry,Logger,Options。

Entry

代码:https://github.com/marmotedu/gopractise-demo/blob/master/log/cuslog/entry.go。

//构造函数func entry(logger *logger) *Entry {  return &Entry{logger: logger, Buffer: new(bytes.Buffer), Map: make(map[string]interface{}, 5)}}/*entry主要方法是write方法,它首先通过e.logger.opt.level 与 level的比较,来判断是否要将日志输出,其中DEBUG最低,FATAL最高,这个从https://github.com/marmotedu/gopractise-demo/blob/master/log/cuslog/options.go#L19中可以看到。*/func (e *Entry) write(level Level, format string, args ...interface{}) {  if e.logger.opt.level > level {    return  }  e.Time = time.Now()  e.Level = level  e.Format = format  e.Args = args  if !e.logger.opt.disableCaller {    if pc, file, line, ok := runtime.Caller(2); !ok {      e.File = "???"      e.Func = "???"    } else {      e.File, e.Line, e.Func = file, line, runtime.FuncForPC(pc).Name()      e.Func = e.Func[strings.LastIndex(e.Func, "/")+1:]    }  }  e.format()  e.writer()  e.release()}

Entry的write方法实现了将它Buffer中的数据,写入到它的logger所配置的output中。

Logger

代码:https://github.com/marmotedu/gopractise-demo/blob/master/log/cuslog/logger.go。

/*创建方法,通过sync.Pool缓存对象,提升性能,initOptions是用于初始化logger的options的各种属性*/func New(opts ...Option) *logger {  logger := &logger{opt: initOptions(opts...)}  logger.entryPool = &sync.Pool{New: func() interface{} { return entry(logger) }}  return logger}
Options

代码:https://github.com/marmotedu/gopractise-demo/blob/68e100ee78a3093e6f2434439e7d4b143b9ebf60/log/cuslog/options.go。

/*opts 参数是一系列的用于设置options属性的函数,比如下面的WithOutput和WithLevel都是这种函数,initOptions会接收这些函数作为输入,对o = &options{}进行设置*/type Option func(*options)func initOptions(opts ...Option) (o *options) {  o = &options{}  for _, opt := range opts {    opt(o)  }  if o.output == nil {    o.output = os.Stderr  }  if o.formatter == nil {    o.formatter = &TextFormatter{}  }  return}func WithOutput(output io.Writer) Option {  return func(o *options) {    o.output = output  }}func WithLevel(level Level) Option {  return func(o *options) {    o.level = level  }}
应用

通过下面的代码,我们看看整个代码是如何串起来的。

// 输出到文件  fd, err := os.OpenFile("test.log", os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)  if err != nil {    log.Fatalln("create file test.log failed")  }  defer fd.Close()  l := cuslog.New(cuslog.WithLevel(cuslog.InfoLevel),    cuslog.WithOutput(fd),    cuslog.WithFormatter(&cuslog.JsonFormatter{IgnoreBasicFields: false}),  )  l.Info("custom log with json formatter")
上面的整体实现是,将"custom log with json formatter"这段字符串写入到指定文件里面1到6行创建并打开文件重点是8行,cuslog.WithLevel(cuslog.InfoLevel)、cuslog.WithOutput(fd)、cuslog.WithFormatter(&cuslog.JsonFormatter{IgnoreBasicFields: false})三个函数调用,返回三个函数func(o*options),这些函数都是Option类型,因为typeOptionfunc(*options)。然后调用cuslog.New,这个函数上面也给出了,它里面通过initOptions依次调用上面的Option函数,对options对象进行设置,然后把options赋给opt, 并创建logger. logger := &logger{opt: initOptions(opts...)}最后调用l.Info("custom log with json formatter")把字符串输出到文件中.整个调用链是l.info===>通过l的pool获取entry===>调用entry的write(InfoLevel, FmtEmptySeparate, args...),在这个write函数里面,只有infoLevel比logger.level优先级大或相等,才输出。并且通过runtime.Caller(2)获取最上层调用info时的,文件名,行号,函数名等信息。因为这个地方有2层嵌套才调用到entry的write,所以runtime.Caller(2)的参数是2总结

上面的代码实现了基本的日志功能,包括日志级别、日志格式配置、输出文件或标准输出的设置。但是一些高级的功能,比如按级别分类输出,Hook能力,结构化日志。这些目前都不支持。

关键词:

下一篇: 最后一页
上一篇: 当前观点:美财政部更新电动汽车税收减免规则,大众、日产、宝马等失去补贴

当前头条:如何使用GO语言从零实现日志包

当前观点:美财政部更新电动汽车税收减免规则,大众、日产、宝马等失去补贴

环球简讯:厄尔尼诺对农产品的影响是……

港媒:在阿富汗问题上,中国积极做事,美国只顾指责

世界热资讯!张兴海朋友圈点赞AITO问界M5智驾版:先试驾先体验再买车!

女子无证骑车遇沉降路段摔倒遭碾压身亡,事故主责认定引争议_环球热议

定档!王鹤棣周也领衔主演的电视连续剧《战火中的青春》4月23日开播 天天短讯

记者:英超对赫拉芬贝赫很有吸引力,他不想再踢一个赛季替补

港股概念追踪 | 国家超算互联网工作正式启动!算力产业链布局愈发清晰(附概念股) 天天实时

移远通信2022年实现营业收入142.30亿元,同比增长26.36%_环球报资讯

别克英朗gt内饰改装图片_别克英朗gt改装图_快资讯

中国清洁能源科技(02379)委任蒋智坚为授权代表|天天快看点

漫威《复仇者联盟 3:无限战争》电影删减了 45 分钟的灭霸剧情

百天16名中管干部被查,传递什么信号? 天天聚看点

湖南经济分析报告

一次元是什么意思_一次元

世纪华通:公司正积极尝试以人工智能技术赋能游戏研发、运营等

天天通讯!市值超400亿!光伏“黑马”正式更名!

每日播报!042期姜太公大乐透预测奖号:单挑一注5+2推荐

金山区气象局发布大风蓝色预警【Ⅳ级/一般】

焦点热议:多部门联合印发推进铁水联运高质量发展行动方案 铁路水路联手 物流降本增效

今日精选:选择大城市逐梦还是小城市安稳

两情若是长久时_两情若是长久时|天天播资讯

押中翻倍AI牛股,私募百万高薪招聘相关人才|当前快播

王者荣耀历史举报怎么看 王者怎么看举报人的记录

今头条!三星于QD-LED市场斥资668亿告别LCD面板

汽车门板焊接机_汽车门板焊接机

员工挪用公款4000元怎么处理

好汉归来!汉马选手经过天鹅路激情澎湃|全球简讯

法国罢工已对法甲比赛造成影响 天天速讯

当前资讯!23041期大乐透晒票:好运气令人羡慕,但是努力更重要

“虹馨三所联动”小程序上线啦!调解纠纷再添新渠道!_当前独家

彻底变天!致命一枪打响,世纪大分裂开始了!-环球新消息

环球看热讯:视频 | 衡阳市雁峰区人武部开展基干民兵救火演练

天天微资讯!定位紧凑型MPV,极狐考拉申报信息曝光

广州买房,千万不要为了买而买!