python logging重复记录日志问题的解决方法


Posted in Python onJuly 12, 2018

日志相关概念

日志是一种可以追踪某些软件运行时所发生事件的方法。软件开发人员可以向他们的代码中调用日志记录相关的方法来表明发生了某些事情。一个事件可以用一个可包含可选变量数据的消息来描述。此外,事件也有重要性的概念,这个重要性也可以被称为严重性级别(level)。

日志的作用

通过log的分析,可以方便用户了解系统或软件、应用的运行情况;如果你的应用log足够丰富,也可以分析以往用户的操作行为、类型喜好、地域分布或其他更多信息;如果一个应用的log同时也分了多个级别,那么可以很轻易地分析得到该应用的健康状况,及时发现问题并快速定位、解决问题,补救损失。

简单来讲就是,我们通过记录和分析日志可以了解一个系统或软件程序运行情况是否正常,也可以在应用程序出现故障时快速定位问题。比如,做运维的同学,在接收到报警或各种问题反馈后,进行问题排查时通常都会先去看各种日志,大部分问题都可以在日志中找到答案。再比如,做开发的同学,可以通过IDE控制台上输出的各种日志进行程序调试。对于运维老司机或者有经验的开发人员,可以快速的通过日志定位到问题的根源。可见,日志的重要性不可小觑。日志的作用可以简单总结为以下3点:

  • 程序调试
  • 了解软件程序运行情况,是否正常
  • 软件程序运行故障分析与问题定位

如果应用的日志信息足够详细和丰富,还可以用来做用户行为分析,如:分析用户的操作行为、类型洗好、地域分布以及其它更多的信息,由此可以实现改进业务、提高商业利益。

发现问题

最近在用Python的logging模块记录日志时,遇到了重复记录日志的问题,第一条记录写一次,第二条记录写两次,第三条记录写三次。。。很头疼,这样记日志可不行。网上搜索到了原因与解决方案:

原因:没有移除handler

解决:在日志记录完之后removeHandler

修改前示例代码:

import logging

def log(message):
 logger = logging.getLogger('testlog')

 streamhandler = logging.StreamHandler()
 streamhandler.setLevel(logging.ERROR)
 formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(name)s - %(message)s')
 streamhandler.setFormatter(formatter)

 logger.addHandler(streamhandler)
 logger.error(message)

if __name__ == '__main__':
 log('hi')
 log('hi too')
 log('hi three')

修改前输出结果:

2016-07-08 09:17:29,740 - ERROR - testlog - hi
2016-07-08 09:17:29,740 - ERROR - testlog - hi too
2016-07-08 09:17:29,740 - ERROR - testlog - hi too
2016-07-08 09:17:29,740 - ERROR - testlog - hi three
2016-07-08 09:17:29,740 - ERROR - testlog - hi three
2016-07-08 09:17:29,740 - ERROR - testlog - hi three

修改后示例代码:

import logging

def log(message):
 logger = logging.getLogger('testlog')

 streamhandler = logging.StreamHandler()
 streamhandler.setLevel(logging.ERROR)
 formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(name)s - %(message)s')
 streamhandler.setFormatter(formatter)

 logger.addHandler(streamhandler)
 logger.error(message)

 # 添加下面一句,在记录日志之后移除句柄
 logger.removeHandler(streamhandler)

if __name__ == '__main__':
 log('hi')
 log('hi too')
 log('hi three')

修改后输出结果:

2016-07-08 09:32:28,206 - ERROR - testlog - hi
2016-07-08 09:32:28,206 - ERROR - testlog - hi too
2016-07-08 09:32:28,206 - ERROR - testlog - hi three

深度解析:

Google之后,大概搞明白了,就是你第二次调用log的时候,根据getLogger(name)里的name获取同一个logger,而这个logger里已经有了第一次你添加的handler,第二次调用又添加了一个handler,所以,这个logger里有了两个同样的handler,以此类推,调用几次就会有几个handler。。

所以这里有以下几个解决办法:

  • 每次创建不同name的logger,每次都是新logger,不会有添加多个handler的问题。(ps:这个办法太笨,不过我之前就是这么干的。。)
  • 像上面一样每次记录完日志之后,调用removeHandler()把这个logger里的handler移除掉。
  • 在log方法里做判断,如果这个logger已有handler,则不再添加handler。
  • 与方法2一样,不过把用pop把logger的handler列表中的handler移除。

下面是方法3与方法4的代码示例:

方法3:

import logging

def log(message):
 logger = logging.getLogger('testlog')

 # 这里进行判断,如果logger.handlers列表为空,则添加,否则,直接去写日志
 if not logger.handlers:
 streamhandler = logging.StreamHandler()
 streamhandler.setLevel(logging.ERROR)
 formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(name)s - %(message)s')
 streamhandler.setFormatter(formatter)
 logger.addHandler(streamhandler)

 logger.error(message)

if __name__ == '__main__':
 log('hi')
 log('hi too')
 log('hi three')

方法4:

import logging

def log(message):
 logger = logging.getLogger('testlog')

 streamhandler = logging.StreamHandler()
 streamhandler.setLevel(logging.ERROR)
 formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(name)s - %(message)s')
 streamhandler.setFormatter(formatter)

 logger.addHandler(streamhandler)

 logger.error(message)

 # 用pop方法把logger.handlers列表中的handler移除,注意如果你add了多个handler,这里需多次pop,或者可以直接为handlers列表赋空值
 logger.handlers.pop()
 # logger.handler = []

if __name__ == '__main__':
 log('hi')
 log('hi too')
 log('hi three')

这几种方法都亲试可行,个人觉得方法3判断更加优雅,你觉得呢?

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
python3使用urllib示例取googletranslate(谷歌翻译)
Jan 23 Python
Python的Django中django-userena组件的简单使用教程
May 30 Python
Django admin实现图书管理系统菜鸟级教程完整实例
Dec 12 Python
Python列表推导式、字典推导式与集合推导式用法实例分析
Feb 07 Python
python装饰器-限制函数调用次数的方法(10s调用一次)
Apr 21 Python
浅谈Series和DataFrame中的sort_index方法
Jun 07 Python
python中enumerate() 与zip()函数的使用比较实例分析
Sep 03 Python
Python上下文管理器用法及实例解析
Nov 11 Python
TensorFlow2.0:张量的合并与分割实例
Jan 19 Python
Python Selenium 设置元素等待的三种方式
Mar 18 Python
Pyecharts 动态地图 geo()和map()的安装与用法详解
Mar 25 Python
Python 中如何使用 virtualenv 管理虚拟环境
Jan 21 Python
python 读写文件,按行修改文件的方法
Jul 12 #Python
Python实现的网页截图功能【PyQt4与selenium组件】
Jul 12 #Python
python基础学习之如何对元组各个元素进行命名详解
Jul 12 #Python
详解Python中的分组函数groupby和itertools)
Jul 11 #Python
Python中的groupby分组功能的实例代码
Jul 11 #Python
python中实现字符串翻转的方法
Jul 11 #Python
Python3.7中安装openCV库的方法
Jul 11 #Python
You might like
php 结果集的分页实现代码
2009/03/10 PHP
MYSQL 小技巧 -- LAST_INSERT_ID
2009/11/24 PHP
深入php处理整数函数的详解
2013/06/09 PHP
浅谈php安全性需要注意的几点事项
2014/07/17 PHP
form中限制文本字节数js代码
2007/06/10 Javascript
js 禁用浏览器的后退功能的简单方法
2008/12/10 Javascript
Extjs学习笔记之七 布局
2010/01/08 Javascript
基于jquery的页面划词搜索JS
2010/09/14 Javascript
js判断undefined类型,undefined,null, 的区别详细解析
2013/12/16 Javascript
js随机生成网页背景颜色的方法
2015/02/26 Javascript
Jquery实现弹性滑块滑动选择数值插件
2015/08/08 Javascript
javascript 使用for循环时该注意的问题-附问题总结
2015/08/19 Javascript
基于Bootstrap重置输入框内容按钮插件
2016/05/12 Javascript
BootStrap Progressbar 实现大文件上传的进度条的实例代码
2016/06/27 Javascript
js 动态给元素添加、移除事件的实现方法
2016/07/19 Javascript
JS控制HTML元素的显示和隐藏的两种方法
2016/09/27 Javascript
Vue精简版风格指南(推荐)
2018/01/30 Javascript
微信小程序商品详情页的底部弹出框效果
2020/11/16 Javascript
Vue SPA单页应用首屏优化实践
2018/06/28 Javascript
vue-router权限控制(简单方式)
2018/10/29 Javascript
JS面向对象编程基础篇(三) 继承操作实例详解
2020/03/03 Javascript
JavaScript适配器模式原理与用法实例详解
2020/03/09 Javascript
使用JS实现鼠标放上图片进行放大离开实现缩小功能
2021/01/27 Javascript
学习python的几条建议分享
2013/02/10 Python
python中实现迭代器(iterator)的方法示例
2017/01/19 Python
对YOLOv3模型调用时候的python接口详解
2019/08/26 Python
通信工程毕业生自荐信
2013/11/01 职场文书
搞笑创意广告语
2014/03/17 职场文书
2014年英语教师工作总结
2014/12/03 职场文书
工会积极分子个人总结
2015/03/03 职场文书
2019消防宣传标语!
2019/07/10 职场文书
大学生军训心得体会5篇
2019/08/15 职场文书
pdf论文中python画的图Type 3 fonts字体不兼容的解决方案
2021/04/24 Python
详解nginx进程锁的实现
2021/06/14 Servers
springboot 启动如何排除某些bean的注入
2021/08/02 Java/Android
Python编程super应用场景及示例解析
2021/10/05 Python