zoukankan      html  css  js  c++  java
  • 自定义logging函数中重复打印问题

    最近碰到了这个问题,很头疼。幸亏找到了这篇文章:http://blog.csdn.net/huilan_same/article/details/51858817

    其实还有一个解决方法,就是直接在模块里面生成一个logger对象,这样加载模块时就只加载一次,不会出现重复调用函数,加载handler的现象。

    用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')
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    修改前输出结果:

    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')
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    修改后输出结果:

    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 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')
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    方法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判断更加优雅,你觉得呢?
  • 相关阅读:
    监控视频长度压缩算法
    获取客户端IP
    常用API接口签名验证参考
    .NET发布的程序代码防止反编译
    SQL Server 获取日期时间并格式化
    SQL Server2008R2可疑状态恢复
    限制网站报错信息暴露在外(客户端可以查看到)
    发布网站时线上网站务必把debug设置false
    IIS上的项目网站关闭Http请求中的Trace和OPTIONS
    使用uploadify上传大文件报 IO error #2038错误的解决方案
  • 原文地址:https://www.cnblogs.com/ohahastudy/p/8195685.html
Copyright © 2011-2022 走看看