博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
python logging 重复写日志问题
阅读量:6951 次
发布时间:2019-06-27

本文共 3111 字,大约阅读时间需要 10 分钟。

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

原因:没有移除handler

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

修改前示例代码:

import loggingdef 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 loggingdef 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。。

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

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

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

方法3:

import loggingdef 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 loggingdef 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')

 

转载于:https://www.cnblogs.com/fmgao-technology/p/9185584.html

你可能感兴趣的文章
我的友情链接
查看>>
如何快速的增加行业经验
查看>>
linux下各个目录的作用说明
查看>>
powershell备份代码和数据库到本地和异地
查看>>
ubuntu 访问dell Idrac卡方法实践
查看>>
MAC OS 如何从Finder中选中目录快捷进入Terminal
查看>>
centos6.3 安装cacti完整实录
查看>>
程序员必备:手把手教你清理烂代码
查看>>
Android项目:使用pulltorefresh开源项目扩展为下拉刷新上拉加载更多的处理方法,监听listview滚动方向...
查看>>
Modernizr.js
查看>>
胖子哥的大数据之路(8)- 数据仓库命名规范
查看>>
程序员,你敢休息一下吗?——真爱生命,远离编程
查看>>
Lync Server 2010详解系列8:Lync移动功能一步到位
查看>>
Debian系统安装xen并创建win2003虚拟机
查看>>
Exchange 2013学习,查看邮件头
查看>>
在Virtualbox虚拟机内外用GoodSync做文件同步的实现
查看>>
我的友情链接
查看>>
ubuntu 远程windows
查看>>
我的友情链接
查看>>
Active Diretory 全攻略(五)--规划和建立组
查看>>