日志是一个框架的重要组成部分,那今天我们一起来看看这部分。
衡量日志库有多个指标,我们今天重点关注两点:简单易用 与 高性能。简单易用是一个日志库能被广泛使用的必要条件,而高性能则是企业级的日志库非常重要的衡量点,也能在源码层面对我们有一定的启发。
v0.3.0:日志库的选型与引入
项目链接 https://github.com/Junedayday/micro_web_service/tree/v0.3.0
目标
选择一个开源的日志组件引入,支持常规的日志打印。
关键技术点
- 三款开源日志库的横向对比
- zap日志库的关键实现
- 关于日志参数的解析
目录构造
1 | --- micro_web_service 项目目录 |
1.三款开源日志库的横向对比
- glog: https://github.com/golang/glog
- logrus: https://github.com/sirupsen/logrus
- zap: https://github.com/uber-go/zap
如果用一次词语分别进行概括三者的特性,我分别会用:glog - 代码极简,logrus - 功能全面, zap - 高性能。经过反复思考,这个框架会选择zap库作为日志引擎的基本组件,主要考量如下:
- 高性能 - 性能是一个日志库很重要的属性,它往往由前期的设计决定,很难通过后面的优化大幅度提高,所以zap的高性能很难被替代;
- 方便封装 - zap设计简单,容易进行二次封装(glog更简洁,相应地就需要更多的封装代码)
- 大厂背书 - zap库被很多大公司引用,作为内部的日志库的底层,再二次开发
- 源码学习 - zap库对性能追求极高,可以作为高性能Go语言代码的分析样例
2.zap日志库的关键实现
最简化的调用
zap日志库的调用很简单,只需要两行代码就能实现初始化:
1 | logger, _ := zap.NewProduction() |
但这样的zap代码存在两个明显弊端:
- 默认输出到控制台上
- 无法保存到指定目录的文件
核心的日志文件实现
我们增加了一定的特性,代码如下:
1 | var ( |
那我们是如何解决上面两个问题的呢?
- 利用
go.uber.org/zap/zapcore
中的开放配置 - 借用了
github.com/natefinch/lumberjack
这个常用的日志切分库,也顺带实现了日志路径的支持
main函数的调用
1 | var logFilePath = flag.String("l", "log/service.log", "log file path") |
至此,我们的日志功能已经基本打通。
3.关于日志参数的解析
日志参数常见的方式分2种,一个是来自flag
的解析,另一个是来自配置文件。
随着我们功能的拓展,日志库肯定会支持越来越复杂的场景。那这个时候用flag
解析的扩展性就会很差,所以,我更推荐在微服务的框架中,用配置文件的方式去加载日志的相关配置。但这种方式会带来一个常见的现象:
程序代码的实现为:先加载配置文件,后加载日志,导致配置文件出错时,无法通过日志来排查,需要用控制台或者进程管理工具协助定位问题。
后续,随着框架的迭代,我会开放出更多的日志参数,目前只放出了一个日志路径的参数作为示例。
后续的两点核心需求
至此,我们添加的代码量并不多,也算成功地实现了一个日志打印的功能。但在实际的工程中,日志模块还需要实现两个比较大的功能:
- 支持Go程序Panic/Error Wrapping风格的多行打印与采集
- 支持分布式TraceId的打印,用来排查微服务调用链路
这两块的内容会结合具体的相关相关技术,会在后续专题中专门分享,请大家重点关注。
总结
zap
库的代码是一个很棒的实现,我会在接下来的Go语言技巧系列中详细分析,欢迎大家进行关注。
至此,我们的框架逐渐成型,接下来我将对GORM
做一个简单的讲解,引入到框架中。
Github: https://github.com/Junedayday/code_reading
Blog: http://junes.tech/
Bilibili: https://space.bilibili.com/293775192
公众号: golangcoding