Python日志打印里logging.getLogger源码分析详解


Posted in Python onJanuary 17, 2021

实践环境

WIN 10

Python 3.6.5

函数说明

logging.getLogger(name=None)

getLogger函数位于logging/__init__.py脚本

源码分析

_loggerClass = Logger
# ...略
 
root = RootLogger(WARNING)
Logger.root = root
Logger.manager = Manager(Logger.root)
 
# ...略
 
def getLogger(name=None):
  """
  Return a logger with the specified name, creating it if necessary.
 
  If no name is specified, return the root logger.
  """
  if name:
    return Logger.manager.getLogger(name)
  else:
    return root

结论:如函数注释所述,如果调用getLogger时,如果没有指定函数参数(即要获取的日志打印器名称)或者参数值不为真,则默认返回root打印器

Logger.manager.getLogger(self, name)源码分析

该函数位于logging/__init__.py脚本

class Manager(object):
  """
  There is [under normal circumstances] just one Manager instance, which
  holds the hierarchy of loggers.
  """
  def __init__(self, rootnode):
    """
    Initialize the manager with the root node of the logger hierarchy.
    """
    self.root = rootnode
    self.disable = 0
    self.emittedNoHandlerWarning = False
    self.loggerDict = {}
    self.loggerClass = None
    self.logRecordFactory = None
 
  def getLogger(self, name):
    """
    Get a logger with the specified name (channel name), creating it
    if it doesn't yet exist. This name is a dot-separated hierarchical
    name, such as "a", "a.b", "a.b.c" or similar.
 
    If a PlaceHolder existed for the specified name [i.e. the logger
    didn't exist but a child of it did], replace it with the created
    logger and fix up the parent/child references which pointed to the
    placeholder to now point to the logger.
    """
    rv = None
    if not isinstance(name, str):
      raise TypeError('A logger name must be a string')
    _acquireLock()
    try:
      if name in self.loggerDict:
        rv = self.loggerDict[name]
        if isinstance(rv, PlaceHolder):
          ph = rv
          rv = (self.loggerClass or _loggerClass)(name)
          rv.manager = self
          self.loggerDict[name] = rv
          self._fixupChildren(ph, rv)
          self._fixupParents(rv)
      else:
        rv = (self.loggerClass or _loggerClass)(name) # _loggerClass = Logger
        rv.manager = self
        self.loggerDict[name] = rv
        self._fixupParents(rv)
    finally:
      _releaseLock()
    return rv

Logger源码分析

_nameToLevel = {
  'CRITICAL': CRITICAL,
  'FATAL': FATAL,
  'ERROR': ERROR,
  'WARN': WARNING,
  'WARNING': WARNING,
  'INFO': INFO,
  'DEBUG': DEBUG,
  'NOTSET': NOTSET,
}
 
# ...略
 
def _checkLevel(level):
  if isinstance(level, int):
    rv = level
  elif str(level) == level:
    if level not in _nameToLevel:
      raise ValueError("Unknown level: %r" % level)
    rv = _nameToLevel[level]
  else:
    raise TypeError("Level not an integer or a valid string: %r" % level)
  return rv
 
# ...略
class PlaceHolder(object):
  """
  PlaceHolder instances are used in the Manager logger hierarchy to take
  the place of nodes for which no loggers have been defined. This class is
  intended for internal use only and not as part of the public API.
  """
  def __init__(self, alogger):
    """
    Initialize with the specified logger being a child of this placeholder.
    """
    self.loggerMap = { alogger : None }
 
  def append(self, alogger):
    """
    Add the specified logger as a child of this placeholder.
    """
    if alogger not in self.loggerMap:
      self.loggerMap[alogger] = None
 
 
 
class Logger(Filterer):
  """
  Instances of the Logger class represent a single logging channel. A
  "logging channel" indicates an area of an application. Exactly how an
  "area" is defined is up to the application developer. Since an
  application can have any number of areas, logging channels are identified
  by a unique string. Application areas can be nested (e.g. an area
  of "input processing" might include sub-areas "read CSV files", "read
  XLS files" and "read Gnumeric files"). To cater for this natural nesting,
  channel names are organized into a namespace hierarchy where levels are
  separated by periods, much like the Java or Python package namespace. So
  in the instance given above, channel names might be "input" for the upper
  level, and "input.csv", "input.xls" and "input.gnu" for the sub-levels.
  There is no arbitrary limit to the depth of nesting.
  """
  def __init__(self, name, level=NOTSET):
    """
    Initialize the logger with a name and an optional level.
    """
    Filterer.__init__(self)
    self.name = name
    self.level = _checkLevel(level)
    self.parent = None
    self.propagate = True
    self.handlers = []
    self.disabled = False
 
  # ... 略

结论:如果调用logging.getLogger()时,有指定日志打印器名称,且名称为真(不为空字符串,0,False等False值),

1)如果名称为不存在的日志打印器名称,则,且参数值为真,但是即要获取的日志打印器名称)或者参数值不为真,则创建一个名为给定参数值的日志打印器,该日志打印器,默认级别默认为NOTSET,disable_existing_loggers配置为False,propagate配置为True。然后在日志打印器字典中记录该名称和日志打印器的映射关系,接着调用 _fixupParents(创建的日志打印器实例)类实例方法--为日志打印器设置上级日志打印器,最后返回该日志打印器。

2)如果名称已存在日志打印器名称,则获取该日志打印器,然后判断日志打印器是否为PlaceHolder类实例,如果是,则创建一个名为所给参数值的日志打印器,同第1)点,该日志打印器,默认级别默认为NOTSET,disable_existing_loggers配置为False,propagate配置为True。然后在日志打印器字典中记录该名称和日志打印器的映射关系,接着调用 _fixupParents(创建的打印器实例)类实例方法,_fixupChildren(PlaceHolder类实例--根据名称获取的日志打印器,新建的日志打印器实例)--为新建日志打印器设置上级日志打印器,为PlaceHolder类实例现有下级PlaceHolder日志打印器实例重新设置上级日志打印器,最后返回该日志打印器。

_fixupParents及_fixupChildren函数源码分析

# _fixupParents
 
# ...略
class Manager(object):
  # ...略
  def _fixupParents(self, alogger):
    """
    Ensure that there are either loggers or placeholders all the way
    from the specified logger to the root of the logger hierarchy.
    """
    name = alogger.name # 获取日志打印器名称
    i = name.rfind(".")
    rv = None # 存放alogger日志打印器的上级日志打印器
    while (i > 0) and not rv: # 如果名称中存在英文的点,并且找到上级日志打印器
      substr = name[:i] # 获取名称中位于最后一个英文的点的左侧字符串(暂且称至为 点分上级)
      if substr not in self.loggerDict: # 如果 点分上级 不存在日志打印器字典中
        self.loggerDict[substr] = PlaceHolder(alogger) # 创建PlaceHolder实例作为 点分上级 对应的日志打印器 # 继续查找点分上级日志打印器 # 注意,这里的PlaceHolder仅是占位用,不是真的打印器,这里为了方便描述,暂且称之为PlaceHolder日志打印器
      else: # 否则
        obj = self.loggerDict[substr] # 获取 点分上级 对应的日志打印器
        if isinstance(obj, Logger): # 如果为Logger实例,如果是,则跳出循环,执行 # 为日志打印器设置上级
          rv = obj
        else: # 否则
          assert isinstance(obj, PlaceHolder) # 断言它为PlaceHolder的实例
          obj.append(alogger) # 把日志打印器添加为点分上级对应的PlaceHolder日志打印器实例的下级日志打印器 执行 # 继续查找点分上级日志打印器
      i = name.rfind(".", 0, i - 1) # 继续查找点分上级日志打印器
    if not rv: # 找不到点分上级、或者遍历完所有点分上级,都没找到上级日志打印器
      rv = self.root # 则 把root日志打印器设置为alogger日志打印器的上级日志打印器
    alogger.parent = rv # 为日志打印器设置上级
 
 
 
  def _fixupChildren(self, ph, alogger):
    """
    Ensure that children of the placeholder ph are connected to the
    specified logger.
    """
    name = alogger.name # 获取日志打印器名称
    namelen = len(name) # 获取日志打印器名称长度
    for c in ph.loggerMap.keys(): # 遍历获取的PlaceHolder日志打印器实例的子级日志打印器
      #The if means ... if not c.parent.name.startswith(nm)
      if c.parent.name[:namelen] != name: # 如果PlaceHolder日志打印器实例名称不以alogger日志打印器名称为前缀,
        alogger.parent = c.parent # 那么,设置alogger日志打印器的上级日志打印器为PlaceHolder日志打印器
        c.parent = alogger # 设置alogger日志打印器为PlaceHolder日志打印器原有下级PlaceHolder日志打印器的上级

结论:日志打印器都是分父子级的,这个父子层级是怎么形成的,参见上述函数代码注解

到此这篇关于Python日志打印里logging.getLogger源码分析详解的文章就介绍到这了,更多相关Python logging.getLogger源码分析内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
python登录QQ邮箱发信的实现代码
Feb 10 Python
python使用urllib2模块获取gravatar头像实例
Dec 18 Python
教你如何将 Sublime 3 打造成 Python/Django IDE开发利器
Jul 04 Python
详解Python中for循环的使用方法
May 14 Python
Python的Tornado框架实现图片上传及图片大小修改功能
Jun 30 Python
详解Python中的动态属性和特性
Apr 07 Python
Python3.6笔记之将程序运行结果输出到文件的方法
Apr 22 Python
Python爬虫基础之XPath语法与lxml库的用法详解
Sep 13 Python
用python3 urllib破解有道翻译反爬虫机制详解
Aug 14 Python
浅谈python元素如何去重,去重后如何保持原来元素的顺序不变
Feb 28 Python
aws 通过boto3 python脚本打pach的实现方法
May 10 Python
pymongo insert_many 批量插入的实例
Dec 05 Python
Python中的面向接口编程示例详解
Jan 17 #Python
Python学习之time模块的基本使用
Jan 17 #Python
python中re模块知识点总结
Jan 17 #Python
史上最详细的Python打包成exe文件教程
Jan 17 #Python
python制作微博图片爬取工具
Jan 16 #Python
python工具——Mimesis的简单使用教程
Jan 16 #Python
Python 内存管理机制全面分析
Jan 16 #Python
You might like
php加密算法之实现可逆加密算法和解密分享
2014/01/21 PHP
php缩放gif和png图透明背景变成黑色的解决方法
2014/10/14 PHP
php使用curl出现Expect:100-continue解决方法
2015/03/03 PHP
typecho插件编写教程(三):保存配置
2015/05/28 PHP
php版银联支付接口开发简明教程
2016/10/14 PHP
PHP使用curl_multi实现并发请求的方法示例
2018/04/29 PHP
Thinkphp5 如何隐藏入口文件index.php(URL重写)
2019/10/16 PHP
不同浏览器对回车提交表单的处理办法
2010/02/13 Javascript
javascript实现数字+字母验证码的简单实例
2014/02/10 Javascript
javascript实现图像循环明暗变化的方法
2015/02/25 Javascript
javascript中Date()函数在各浏览器中的显示效果
2015/06/18 Javascript
Bootstrap进度条实现代码解析
2017/03/07 Javascript
原生JS实现图片网格式渐显、渐隐效果
2017/06/05 Javascript
vue使用Axios做ajax请求详解
2017/06/07 Javascript
vue开发调试神器vue-devtools使用详解
2017/07/13 Javascript
nginx配置React静态页面的方法教程
2017/11/03 Javascript
vue发送websocket请求和http post请求的实例代码
2019/07/11 Javascript
jQuery实现可编辑的表格
2019/12/11 jQuery
vue-property-decorator用法详解
2019/12/12 Javascript
[01:12]DOTA2次级职业联赛 - Newbee.Y 战队宣传片
2014/12/01 DOTA
[46:20]CHAOS vs Alliacne 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/16 DOTA
简单的Apache+FastCGI+Django配置指南
2015/07/22 Python
python 返回列表中某个值的索引方法
2018/11/07 Python
对python当中不在本路径的py文件的引用详解
2018/12/15 Python
Django url,从一个页面调到另个页面的方法
2019/08/21 Python
解决Python中回文数和质数的问题
2019/11/24 Python
Python异常继承关系和自定义异常实现代码实例
2020/02/20 Python
Gap加拿大官网:Gap Canada
2017/08/24 全球购物
中专毕业个人的自荐信格式
2013/09/21 职场文书
大学生个人简历自我评价
2013/11/16 职场文书
优秀应届生求职信
2014/06/16 职场文书
无财产离婚协议书范本
2014/10/28 职场文书
财务负责人岗位职责
2015/02/03 职场文书
奥巴马开学演讲观后感
2015/06/12 职场文书
python实战之90行代码写个猜数字游戏
2021/04/22 Python
Python中Schedule模块使用详解 周期任务神器
2022/04/19 Python