python的logging模块提供了标准的日志接口,你可以通过它存储各种格式的日志,主要用于输出运行日志,可以设置输出日志的等级、日志保存路径、日志文件回滚等。

一、logging日志框架

主要包括四部分:

  • Loggers: 可供程序直接调用的接口,app通过调用提供的api来记录日志
  • Handlers: 决定将日志记录分配至正确的目的地
  • Filters:对日志信息进行过滤, 提供更细粒度的日志是否输出的判断
  • Formatters: 制定最终记录打印的格式布局

1、loggers

loggers 就是程序可以直接调用的一个日志接口,可以直接向logger写入日志信息。logger并不是直接实例化使用的,而是通过logging.getLogger(name)来获取对象,事实上logger对象是单例模式,logging是多线程安全的,也就是无论程序中哪里需要打日志获取到的logger对象都是同一个。但是不幸的是logger并不支持多进程,这个在后面的章节再解释,并给出一些解决方案。

【注意】loggers对象是有父子关系的,当没有父logger对象时它的父对象是root,当拥有父对象时父子关系会被修正。举个例子,logging.getLogger(“abc.xyz”) 会创建两个logger对象,一个是abc父对象,一个是xyz子对象,同时abc没有父对象,所以它的父对象是root。但是实际上abc是一个占位对象(虚的日志对象),可以没有handler来处理日志。但是root不是占位对象,如果某一个日志对象打日志时,它的父对象会同时收到日志,所以有些使用者发现创建了一个logger对象时会打两遍日志,就是因为他创建的logger打了一遍日志,同时root对象也打了一遍日志。

2、Handlers

Handlers 将logger发过来的信息进行准确地分配,送往正确的地方。举个栗子,送往控制台或者文件或者both或者其他地方(进程管道之类的)。它决定了每个日志的行为,是之后需要配置的重点区域。

每个Handler同样有一个日志级别,一个logger可以拥有多个handler也就是说logger可以根据不同的日志级别将日志传递给不同的handler。当然也可以相同的级别传递给多个handlers这就根据需求来灵活的设置了。

3、Filters

Filters 提供了更细粒度的判断,来决定日志是否需要打印。原则上handler获得一个日志就必定会根据级别被统一处理,但是如果handler拥有一个Filter可以对日志进行额外的处理和判断。例如Filter能够对来自特定源的日志进行拦截or修改甚至修改其日志级别(修改后再进行级别判断)。

logger和handler都可以安装filter甚至可以安装多个filter串联起来。

4、Formatters

Formatters 指定了最终某条记录打印的格式布局。Formatter会将传递来的信息拼接成一条具体的字符串,默认情况下Format只会将信息%(message)s直接打印出来。Format中有一些自带的LogRecord属性可以使用,如下表格:

python 配置全局日志模块 python 日志框架_日志框架

二、代码实例

1、logger.py文件

import logging
import logging.handlers
import logging.config
import os

class InfoFilter(logging.Filter):
    def filter(self, record):
        if record.levelno == logging.INFO:
            return super().filter(record)
        else:
            return 0
            
#定义日志输出地址根目录
log_dir = './logs'
#定义日志框架
log_dict = {
    'version': 1,
    'disable_existing_loggers': False,

    'formatters': {
        'standard': {
            'format': '%(asctime)s | %(process)d | %(levelname)s | %(filename)s | %(funcName)s | %(lineno)d | %(message)s'
        }
    },

    'filters': {
        'info_filter': {
            '()': InfoFilter,
        },
    },

    'handlers': {
        'info': {
            'level': 'INFO',
            'class': 'logging.handlers.TimedRotatingFileHandler',
            'filename': os.path.join(log_dir, 'info.log'),
            'when': 'H',
            'formatter': 'standard',
            'filters': ['info_filter']
        },
        'error': {
            'level': 'ERROR',
            'class': 'logging.handlers.TimedRotatingFileHandler',
            'filename': os.path.join(log_dir, 'error.log'),
            'when': 'H',
            'formatter': 'standard'
        },
    },

    'loggers': {
        'info_logger': {
            'handlers': ['info'],
            'level': 'INFO'
        },
        'error_logger': {
            'handlers': ['error'],
            'level': 'ERROR'
        },
    },
}
#配置日志框架
logging.config.dictConfig(log_dict)
#创建logger对象,在调用getLogger时要提供Logger的名称
info_logger = logging.getLogger('info_logger')
error_logger = logging.getLogger('error_logger')

2、使用

from .logger import info_logger, error_logger
info_logger.info("需要输出的日志内容信息")
error_logger.error("需要输出的错误信息")  #一般与try...except搭配使用

3、运行后下面文件会记录对应日志内容

python 配置全局日志模块 python 日志框架_python 配置全局日志模块_02

参考文献:
1、以打印日志为荣之logging模块详细使用