改进log4go的一些设想
log4go 的 4.0.2 版本(https://github.com/ccpaging/log4go/tree/4.0.2)发布以后,
看了看别的 go 语言日志文件设计。发现了一篇好文:
log4go 和 logrus 的对比与分析
https://www.doraemonext.com/archives/783.html
顺藤摸瓜,找了一窝关于日志的设计。链接如下(含老的链接):
- https://github.com/alecthomas/log4go/
 这是log4go项目的“鼻祖”
- https://github.com/ngmoco/timber
 实现了结构化,写文件缓冲,热配置等。把log4go重构的面目全非
- https://github.com/siddontang/go/tree/master/log
- https://github.com/sirupsen/logrus
- https://github.com/YoungPioneers/blog4go
- https://github.com/YoungPioneers/blog4go-benchmark 各种 go log 的benchmark对比
- https://github.com/cihub/seelog
异步写入日志
log4go 的特点之一是异步写入。格式化日志记录、写入文件、转储日志等,都会消耗 CPU 的时间,并可能因为错误处理而阻塞主线程。
但日志系统仅仅是一个辅助功能,所以,保证主线程的高效运行是首先要达到的设计要求。异步写入是可行的方案之一。
自扩展日志接口
其实,log4go 是支持类似 logrus 的扩展特性的。
正好纠结于 color text term log 的设计如何处理的问题……因为这个功能使用了第三方包。放在log4go里增加了它的依赖性。但这确实又是我特别特别喜欢的一个功能。
不如把 color text term log 做成扩展日志接口。说干就干……
先搞清楚 log4go 中可用的扩展接口:
type LogWriter interface {
	LogWrite(rec *LogRecord)
	// This should clean up anything lingering about the LogWriter, as it is called before
	// the LogWriter is removed.  LogWrite should not be called after Close.
	Close()
}
type Filter struct {
	Level Level
	rec 	chan *LogRecord	// write queue
	closed 	bool	// true if Socket was closed at API level
	LogWriter
}
type Logger map[string]*Filter
func (log Logger) AddFilter(name string, lvl Level, writer LogWriter) Logger {
	log[name] = NewFilter(lvl, writer)
	return log
}
扩展程序只要做:
- NewXXXLogWrite,初始化扩展要使用的资源。
- LogWrite(rec *LogRecord),输出日志记录
- 在Close()中关闭或释放资源
- 在应用程序中调用 AddFilter把新的日志扩展加入到log4go日志结构中
大功告成了。
其中,Add filter name 是 Logger map 的索引关键字,log4go 使用了:
"stdout", "file", "syslog"
如果新加的 Filter 的关键字已存在,log4go(4.0.2以后的版本)将自动关闭原来的,再增加新的。代码如下:
func (log Logger) AddFilter(name string, lvl Level, writer LogWriter) Logger {
	if filt, isExist := log[name]; isExist {
		filt.Close()
		delete(log, name)
	}
	log[name] = NewFilter(lvl, writer)
	return log
}
借助扩展接口,log4go的日志记录可以采用任何你希望的封装格式,例如 xml 和 json,这是已经实现的。
以后还可以扩展csv(使日志文件导入到Excel中)或者json封装的message。
可扩展的日志接口包括:
- Send error messages as a mail 
- Make TCP/UDP server and let client pull the messages 
- websocket 
- nanomsg pub/sub 
- Store log messages in MySQL 
自扩展日志配置接口
log4go 4.0.2 支持 xml 和 json 配置。日志文件的配置有三种方式:
- 在应用程序中配置
- 单独的配置文件
- 存于主程序配置文件中
日志系统作为一个辅助功能,常常面临的是第三种情况。而配置文件的格式多种多样。例如:
windows ini, linux config, json, xml ...
郁闷。log4go 不应当去支持所有的配置文件格式,而是提供接口,让用户可以根据自己的主程序的设计需要,自行扩展。
也许应该把 xml 和 json 配置文件支持都以扩展配置文件接口的方式实现,而不是跟 log4go 的主程序捆绑在一起。
文件日志的写缓冲
已经测试了两层缓冲写文件。
第一层是格式化日志记录,一个单独的go routine,另一个写文件,边格式化记录边写文件,消耗降低了40%。
第二层是用bufio。达到一定的缓冲数量如4k、8k,一次写文件。消耗降低了80%。
通过判断Channel中的记录长度来决定系统何时空闲。当长度为0时,后续没有新的日志记录,做一次Flush()。
这种方案简单。
另外加上 rotate 的优化,效率提高了5倍。
BenchmarkFileLog-4                        200000             10675 ns/op
BenchmarkFileNotLogged-4                20000000               106 ns/op
BenchmarkFileUtilLog-4                    200000             10660 ns/op
BenchmarkFileUtilNotLog-4                5000000               239 ns/op
BenchmarkCacheFileLog-4                  1000000              2191 ns/op
BenchmarkCacheFileNotLogged-4           20000000               106 ns/op
BenchmarkCacheFileUtilLog-4               500000              3680 ns/op
BenchmarkCacheFileUtilNotLog-4           5000000               240 ns/op
Rotate 的改进设想
log4go 自带 rotate。
linux 系统本来是有 logrotate 的,用 cron 定时执行。非常棒的设计。
简单说,就是写日志文件归写日志文件,不要去做任何转储的判断。程序员可根据系统的实际运行情况,
自行设置转储的时间间隔。转储时:
- 加锁。使 log4go 暂时停止写日志,这可能是在 linux 系统中 log4go 没有使用 logrotate 的原因之一。 
- 对当前日志文件进行处理。 
- 解锁。尽快恢复 log4go,继续写日志到当前日志文件。 
- 另开 go routine 对历史日志文件进行处理。 
好吧。暂时就想到这么多了。很多有趣的工作正在进行……
再次感谢 doraemonext@gmail.com 童鞋的好文:log4go 和 logrus 的对比与分析
请关注:
https://github.com/ccpaging/log4go
改进log4go的一些设想的更多相关文章
- 团队作业7---Alpha冲刺值事后诸葛
		一.设想和目标 1.我们的软件要解决什么问题? 解决教师和助教对实验报告查重的问题,拥有两个用户:1.教师或助教:查看学生实验报告的重复率:4.学生:上传实验报告. 2.是否定义得很清楚?是否对典型用 ... 
- [软工顶级理解组] Beta阶段事后分析
		目录 设想和目标 计划 资源 变更管理 设计/实现 测试/发布 团队的角色,管理,合作 总结 质量提高 会议截图 设想和目标 我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰 ... 
- log4go的一些改进设想
		log4go 的 4.0.2 版本(https://github.com/ccpaging/log4go/tree/4.0.2)发布以后, 看了看别的 go 语言日志文件设计.发现了一篇好文: log ... 
- log4go的日志滚动处理——适应生产环境的需要
		日志处理有三类使用环境,开发环境DE,测试环境TE,生产环境PE. 前两类可以看成是一类,重要的是屏幕显示--termlog.生产环境中主要用的是socklog 和 filelog,即网络传输日志和文 ... 
- log4go的日志滚动处理——生产环境的适配
		日志处理有三类使用环境,开发环境DE,测试环境TE,生产环境PE. 前两类可以看成是一类,重要的是屏幕显示--termlog.生产环境中主要用的是socklog 和 filelog,即网络传输日志和文 ... 
- homework_06 围棋程序改进
		1) 把程序编译通过, 跑起来. 读懂程序,在你觉得比较难懂的地方加上一些注释,这样大家就能比较容易地了解这些程序在干什么. 把正确的 playPrev(GoMove) 的方法给实现了. 注释见Git ... 
- 企业级应用架构(三)三层架构之数据访问层的改进以及测试DOM的发布
		在上一篇我们在宏观概要上对DAL层进行了封装与抽象.我们的目的主要有两个:第一,解除BLL层对DAL层的依赖,这一点我们通过定义接口做到了:第二,使我们的DAL层能够支持一切数据访问技术,如Ado.n ... 
- 数据访问层的改进以及测试DOM的发布
		数据访问层的改进以及测试DOM的发布 在上一篇我们在宏观概要上对DAL层进行了封装与抽象.我们的目的主要有两个:第一,解除BLL层对DAL层的依赖,这一点我们通过定义接口做到了:第二,使我们的DAL层 ... 
- log4go的全局封装Wrapper和标准log库函数的兼容
		方便易用的全局函数 大多数时候,只不过是写一个简单的测试程序.例如: package main import ( "log" ) func main(){ log.Fatal(&q ... 
随机推荐
- JSONP不支持循环调用
			问题描述 在jquery或zepto下,循环调用同一个jsonp ;i<;i++){ $.ajax({ url:'https://m.suning.com/authStatus?callback ... 
- jquery each 遍历
			在jquery中,遍历对象和数组,经常会用到$().each和$.each(),两个方法. $().each 在dom处理上面用的较多.如果页面有多个input标签类型为checkbox,对于这时用$ ... 
- HTML----网页基础和基本标签
			网页分类: 1.静态网页:所有内容全写死,都写在源代码中,若修改必须修改源代码,后缀为.html或htm 2.动态网页:内容大部分来自于数据库,可以修改,后缀为.aspx(c#).jsp(java). ... 
- std::forward_list
			forward_list相比list来说空间利用率更好,与list一样不支持随机访问,若要访问除头尾节点的其他节点则时间复杂度为线性. 在forward_list成员函数里只能访问头节点以及向头节点插 ... 
- 字符串的拼接python
			数字可以强制转换为字符串,但是字符串不能强制转换为数字(会报错) a='abcs' b='dsys' 方法一.a+b 最low的一个方法,因为每+一次内存增加一次 方法二.print '%s%s'%( ... 
- python http长连接客户端
			背景: 线上机器,需要过滤access日志,发送给另外一个api 期初是单进程,效率太低,改为多进程发送后,查看日志中偶尔会出现异常错误(忘记截图了...) 总之就是端口不够用了报错 原因: 每一条日 ... 
- iOS开发之常用资讯类App的分类展示与编辑的完整案例实现(Swift版)
			上篇博客我们聊了<资讯类App常用分类控件的封装与实现(CollectionView+Swift3.0)>,今天的这篇博客就在上篇博客的基础上做些东西.做一个完整的资讯类App中的分类展示 ... 
- solr5Ik分词2
			<!--IK分词器--><fieldType name="text_ik" class="solr.TextField"><ana ... 
- MATLAB仿真中连续和离散的控制器有何区别?
			matlab系统同时提供连续和离散的控制器和对象的目的是:在降低用户使用复杂程度的同时提高仿真精度.仿真速度和应用的广泛性. 仿真步长和求解精度的概念对于理解这个问题至关重要. 首先是步长,步长和求解 ... 
- oracle表的简单操作
			版权声明:本文为博主原创文章,转载时请注明原文链接. 1.创建表 ) ) not null,primary key(num)); 创建了一个两个字段的表,num和name,都设置为非空,num设为主键 ... 
