先上解决方案:(来自:https://blog.csdn.net/xuan_010/article/details/118566749 )

loger = logging.getLogger(name)
loger.propagate = False

原因:

1、依据logging包官方的注释,日志记录会向上传播到他的父节点也就是root logger
2、当root logger 同时也被添加了屏幕输出handler的情况,日志就会输出第二次

3、root logger 若被多次添加屏幕输出handler,同一日志就会多次输出。

另外:如果有一个专门用来创建新loger对象的函数(例如 def newLoger(name):),那么要注意避免root 多次addHandler。(可以通过判断语句,在root 添加 handler之前就直接return。)

或者handler尽量不要写在函数里面,可以在函数外层创建一个全局变量保存handler,避免每次调用newLoger()都创建一个新的handler对象,造成root 添加了很多个hanlder。(root 多次add同一个handler是不会拥有多个handler的,但add不同的handler就会拥有多个handler)

验证:

import logging,sys,os
def newLoger(name):loger = logging.getLogger(name)loger.setLevel(logging.INFO)logging_format = logging.Formatter("[%(name)s][%(asctime)s]: %(message)s","%H:%M:%S")   # "%Y-%m-%d %H:%M:%S"handler = logging.StreamHandler(sys.stdout)handler.setFormatter(logging_format)loger.addHandler(handler)return logerrootLoger = logging.getLogger()
print(f'rootLoger 的 handler 数量:{len(rootLoger.handlers)}')tom =newLoger('')      # 此时变量tom指向[root loger], 然后给[root Loger]添加了一个handler
print(f'root 的 handler 数量:{len(rootLoger.handlers)}')tom =newLoger('tom')    # 此时变量tom指向一个名为tom的loger, 而[root loger]没有变化
print(f'root 的 handler 数量:{len(rootLoger.handlers)}')jack =newLoger('')      # 此时变量jack指向[root loger], 然后给[root Loger]添加了一个handler
print(f'tom 的 handler 数量:{len(tom.handlers)}')
print(f'root 的 handler 数量:{len(rootLoger.handlers)}')jack =newLoger('jack')      # 此时变量tom指向一个名为tom的loger, 而[root loger]没有变化
print(f'tom 的 handler 数量:{len(tom.handlers)}')
print(f'jack 的 handler 数量:{len(tom.handlers)}')
print(f'root 的 handler 数量:{len(rootLoger.handlers)}')

折腾过程:

代码中要对多个用户的资料进行管理。为了让每个用户单独一个日志文件。使用了logging 。

在屏幕输出的时候却遇到怪异的情况:每引用一次root logger【根打印器】之后,所有打印器的输出都会增加一个重复。(创建logger时没有指定名字 logger = logging.getLogger(),就会返回root loger ,参见:Python 日志打印之logging.getLogger源码分析 - 授客 - 博客园)

感觉就是 logger = logging.getLogger() 不是简单的返回root loger,而是创建一个新的root loger,经过多次  logger = logging.getLogger() 后,内存里面就有很多个 root logger。并且当有名字的logger输出的时候,这些root logger全部都一并跑出来也输出一番。

import logging,sys,os
import timedef newLoger(name):loger = logging.getLogger(name)loger.setLevel(logging.INFO)logging_format = logging.Formatter("[%(name)s][%(asctime)s]: %(message)s","%H:%M:%S")   # "%Y-%m-%d %H:%M:%S"handler = logging.StreamHandler(sys.stdout)     # 屏幕输出loghandler.setFormatter(logging_format)loger.addHandler(handler)return logerclass Player:def __init__(self,account):self.account: str = accountself.log = newLoger(self.account)def hi(self):self.log.info(f'hi,{self.log.name}')time.sleep(1)tom = Player('tom')
tom.hi()jack = Player('')
jack.hi()jack = Player('jack')
jack.hi()mike = Player('')
mike.hi()tom.hi()

输出:

===================

以下摘自 关于logging的那些坑_weixin_30888413的博客-CSDN博客

任何自定义的logger对象都是rootLogger的子对象,

而当你使用自定义的logger记录日志的时候,它会从子到父,依次执行所有的handler来输出,

Logging 之root 打印器来捣乱相关推荐

  1. 5.6. Spring boot with Logging

    一般的日志需求可以通过配置 application.properties实现,如果仍不能满足可以使用 logback.xml 配置日志. logging.file=target/spring.log ...

  2. python多个日志模块怎么隔离_在多个模块中使用Python日志logging

    最好的做法是在每个模块中都有一个像这样定义的logging器: import logging logger = logging.getLogger(__name__) 靠近模块的顶部,然后在模块的其他 ...

  3. Python的日志模块logging的使用

    Python的日志模块logging的使用 1 logging模块介绍 2 logging 简单示例 3 logging.basicConfig() 4 logging组件 4.1 Logger类 4 ...

  4. python三十九:logging模块

    import logginglogging.basicConfig(level=logging.DEBUG # 设置日志级别, 默认为 WARNING,filename="logger.lo ...

  5. Failed to bind properties under ‘logging.level‘ to java.util.Map java.lang.String, java.lang.String

    springBoot 2.0 添加 日志级别,启动报错: logging:level: info 修改成 logging:level:root: info

  6. 操作系统服务:logging日志记录模块

    许多应用程序中都会有日志模块,用于记录系统在运行过程中的一些关键信息,以便于对系统的运行状况进行跟踪. 在.NET平台中,有非常著名的第三方开源日志组件log4net,c++中,有人们熟悉的log4c ...

  7. python logging日志模块的使用

    1.日志级别 日志一共分成5个等级,从低到高分别是:DEBUG ,INFO, WARNING ,ERROR, CRITICAL. DEBUG:详细的信息,通常只出现在诊断问题上 INFO:确认一切按预 ...

  8. Python模块:日志输出—logging模块

    1. logging介绍 Python的logging模块提供了通用的日志系统,可以方便第三方模块或者是应用使用.这个模块提供不同的日志级别,并可以采用不同的方式记录日志,比如文件,HTTP GET/ ...

  9. Python 内置模块之 logging

    日志的级别和适用情况 级别 适用情况 DEBUG 详细信息,通常只在诊断问题时对其感兴趣 INFO 确认工作正常 WARNING 表示发生了意料之外的事或者在不远的将来会有问题(比如磁盘空间低).软件 ...

最新文章

  1. Matlab数据的可视化 -- 线性图函数plot
  2. anaconda python降级_anaconda 使用的一些体验与困惑
  3. Hadoop HBase概念学习系列之HBase的Shell(步骤非常清晰)(二十四)
  4. python生成器和迭代器区别_生成器、迭代器的区别?
  5. On the Difference Between Orthogonal Matching Pursuit and Orthogonal Least Squares
  6. hmac hmac.new_使用HMAC(Play 2.0)保护REST服务
  7. Python-杨辉三角
  8. 和信虚拟终端的全面部署-虚拟终端网络工程实施
  9. 树状数组---Squared Permutation
  10. MVVM最佳解读和实践
  11. 中国云计算进入全球通时代 阿里云四大海外数据中心相继开服
  12. 数据分析:使用Imblearn处理不平衡数据(过采样、欠采样)
  13. Qt_MinGW编译二维码生成库Zint及使用
  14. 2012服务器系统有什么版本的,Windows server 2012操作系统有哪几个版本
  15. 2018语言排行榜php,世界编程语言排行榜_TIOBE:2019年12月全球编程语言排行榜
  16. FPGA之OV7725摄像头采集与VGA显示实验--3--摄像头配置模块实现(Verilog代码)
  17. VALSE2019总结(6)-年度总结-GAN
  18. 【Python工具】Python实现一款支持各大平台的视频下载器 | 附源码
  19. [附源码]Python计算机毕业设计动物保护资讯推荐网站
  20. opencv 照片动漫风格

热门文章

  1. 我用Python模拟了谷爱凌的凌空一跃
  2. 怎样降低计算机屏幕亮度,如何调低电脑屏幕亮度【解决步骤】
  3. 因为取了个快递我搞懂了五种网络IO模型
  4. Spyder 中 Reloaded modules 错误的解决方法
  5. vns基本的变邻域搜索算法
  6. android手机间的通讯,(一)Android 两部手机经过UDP在局域网内通讯
  7. UBUNTU 22.04 使用 SUNSHINE 和 MOONLIGHT 进行串流
  8. python学习之人民币兑美元之间的转换
  9. CSS中Margin的注意问题
  10. SHT30温湿度模块使用