当前位置:首页 > 技术文章 > 编程技术 > Python

python教程:解决logging重复写日志问题

发布时间:2017-12-22 19:31:44 作者:huilan_same 来源:网络 点击量:

python教程:解决logging重复写日志问题

用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判断更加优雅,你觉得呢?

本文收集整理于: http://blog.csdn.net/huilan_same/article/details/51858817  感谢作者的分享!

本文链接:http://www.topjishu.net/article/89.html,如有转载,请注明出处!


关键词:python教程,logging重复写日志


版权声明:TOP技术学院所发布内容部分为原创发布,本着技术分享的精神,也有部分收集整理来自互联网,对于来源明确的内容,会严格注明出处。收集整理的内容本平台不拥有所有权,也不承担相关法律责任。如果您发现本平台中有涉嫌抄袭的内容,可以联系管理员进行举报,并提供相关证据,一经查实,本平台将立刻删除涉嫌侵权内容。
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表