做为开发者,咱们能够经过如下3种方式来配置logging:html
fileConfig()
函数来读取该文件的内容;dictConfig()
函数;须要说明的是,logging.basicConfig()
也属于第一种方式,它只是对loggers, handlers和formatters的配置函数进行了封装。另外,第二种配置方式相对于第一种配置方式的优势在于,它将配置信息和代码进行了分离,这一方面下降了日志的维护成本,同时还使得非开发人员也可以去很容易地修改日志配置。python
代码以下:socket
# 建立一个日志器logger并设置其日志级别为DEBUG logger = logging.getLogger('simple_logger') logger.setLevel(logging.DEBUG) # 建立一个流处理器handler并设置其日志级别为DEBUG handler = logging.StreamHandler(sys.stdout) handler.setLevel(logging.DEBUG) # 建立一个格式器formatter并将其添加处处理器handler formatter = logging.Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s") handler.setFormatter(formatter) # 为日志器logger添加上面建立的处理器handler logger.addHandler(handler) # 日志输出 logger.debug('debug message') logger.info('info message') logger.warn('warn message') logger.error('error message') logger.critical('critical message')
运行输出:函数
2017-05-15 11:30:50,955 - simple_logger - DEBUG - debug message 2017-05-15 11:30:50,955 - simple_logger - INFO - info message 2017-05-15 11:30:50,955 - simple_logger - WARNING - warn message 2017-05-15 11:30:50,955 - simple_logger - ERROR - error message 2017-05-15 11:30:50,955 - simple_logger - CRITICAL - critical message
如今咱们经过配置文件的方式来实现与上面一样的功能:debug
# 读取日志配置文件内容 logging.config.fileConfig('logging.conf') # 建立一个日志器logger logger = logging.getLogger('simpleExample') # 日志输出 logger.debug('debug message') logger.info('info message') logger.warn('warn message') logger.error('error message') logger.critical('critical message')
配置文件logging.conf
内容以下:日志
[loggers] keys=root,simpleExample [handlers] keys=fileHandler,consoleHandler [formatters] keys=simpleFormatter [logger_root] level=DEBUG handlers=fileHandler [logger_simpleExample] level=DEBUG handlers=consoleHandler qualname=simpleExample propagate=0 [handler_consoleHandler] class=StreamHandler args=(sys.stdout,) level=DEBUG formatter=simpleFormatter [handler_fileHandler] class=FileHandler args=('logging.log', 'a') level=ERROR formatter=simpleFormatter [formatter_simpleFormatter] format=%(asctime)s - %(name)s - %(levelname)s - %(message)s datefmt=
运行输出:code
2017-05-15 11:32:16,539 - simpleExample - DEBUG - debug message 2017-05-15 11:32:16,555 - simpleExample - INFO - info message 2017-05-15 11:32:16,555 - simpleExample - WARNING - warn message 2017-05-15 11:32:16,555 - simpleExample - ERROR - error message 2017-05-15 11:32:16,555 - simpleExample - CRITICAL - critical message
fileConfig()
函数的说明:该函数其实是对configparser
模块的封装,关于configparser
模块的介绍请参考<
该函数定义在loging.config模块下:xml
logging.config.fileConfig(fname, defaults=None, disable_existing_loggers=True)
上面提到过,fileConfig()
函数是对ConfigParser/configparser
模块的封装,也就是说fileConfig()
函数是基于ConfigParser/configparser
模块来理解日志配置文件的。换句话说,fileConfig()
函数所能理解的配置文件基础格式是与ConfigParser/configparser
模块一致的,只是在此基础上对文件中包含的section
和option
作了一下规定和限制,好比:htm
1)配置文件中必定要包含loggers
、handlers
、formatters
这些section,它们经过keys
这个option来指定该配置文件中已经定义好的loggers、handlers和formatters,多个值之间用逗号分隔;另外loggers
这个section中的keys必定要包含root这个值;
2)loggers
、handlers
、formatters
中所指定的日志器、处理器和格式器都须要在下面以单独的section进行定义。seciton的命名规则为[logger_loggerName]
、[formatter_formatterName]
、[handler_handlerName]
3)定义logger的section必须指定level
和handlers
这两个option,level
的可取值为DEBUG
、INFO
、WARNING
、ERROR
、CRITICAL
、NOTSET
,其中NOTSET
表示全部级别的日志消息都要记录,包括用户定义级别;handlers
的值是以逗号分隔的handler名字列表,这里出现的handler必须出如今[handlers]这个section中,而且相应的handler必须在配置文件中有对应的section定义;
4)对于非root logger来讲,除了level
和handlers
这两个option以外,还须要一些额外的option,其中qualname
是必须提供的option,它表示在logger层级中的名字,在应用代码中经过这个名字获得logger;propagate
是可选项,其默认是为1,表示消息将会传递给高层次logger的handler,一般咱们须要指定其值为0,这个能够看下下面的例子;另外,对于非root logger的level若是设置为NOTSET,系统将会查找高层次的logger来决定此logger的有效level。
5)定义handler的section中必须指定class
和args
这两个option,level
和formatter
为可选option;class
表示用于建立handler的类名,args
表示传递给class
所指定的handler类初始化方法参数
,它必须是一个元组(tuple)的形式,即使只有一个参数值也须要是一个元组的形式;level
与logger中的level同样,而formatter
指定的是该处理器所使用的格式器,这里指定的格式器名称必须出如今formatters
这个section中,且在配置文件中必需要有这个formatter的section定义;若是不指定formatter则该handler将会以消息自己做为日志消息进行记录,而不添加额外的时间、日志器名称等信息;
6)定义formatter的sectioin中的option都是可选的,其中包括format
用于指定格式字符串,默认为消息字符串自己;datefmt
用于指定asctime的时间格式,默认为'%Y-%m-%d %H:%M:%S'
;class
用于指定格式器类名,默认为logging.Formatter;
说明:
配置文件中的
class
指定类名时,该类名能够是相对于logging模块的相对值,如:FileHandler
、handlers.TimeRotatingFileHandler
;也能够是一个绝对路径值,经过普通的import机制来解析,如自定义的handler类mypackage.mymodule.MyHandler
,可是mypackage须要在Python可用的导入路径中--sys.path。
咱们把logging.conf
中simpleExample这个handler定义中的propagate属性值改成1,或者删除这个option(默认值就是1):
[logger_simpleExample] level=DEBUG handlers=consoleHandler qualname=simpleExample propagate=1
如今来执行一样的代码:
# 读取日志配置文件内容 logging.config.fileConfig('logging.conf') # 建立一个日志器logger logger = logging.getLogger('simpleExample') # 日志输出 logger.debug('debug message') logger.info('info message') logger.warn('warn message') logger.error('error message') logger.critical('critical message')
咱们会发现,除了在控制台有输出信息时候,在logging.log文件中也有内容输出:
2017-05-15 16:06:25,366 - simpleExample - ERROR - error message 2017-05-15 16:06:25,367 - simpleExample - CRITICAL - critical message
这说明simpleExample这个logger在处理完日志记录后,把日志记录传递给了上级的root logger再次作处理,全部才会有两个地方都有日志记录的输出。一般,咱们都须要显示的指定propagate的值为0,防止日志记录向上层logger传递。
如今,咱们试着用一个没有在配置文件中定义的logger名称来获取logger:
# 读取日志配置文件内容 logging.config.fileConfig('logging.conf') # 用一个没有在配置文件中定义的logger名称来建立一个日志器logger logger = logging.getLogger('simpleExample1') # 日志输出 logger.debug('debug message') logger.info('info message') logger.warn('warn message') logger.error('error message') logger.critical('critical message')
运行程序后,咱们会发现控制台没有任何输出,而logging.log文件中又多了两行输出:
2017-05-15 16:13:16,810 - simpleExample1 - ERROR - error message 2017-05-15 16:13:16,810 - simpleExample1 - CRITICAL - critical message
这是由于,当一个日志器没有被设置任何处理器是,系统会去查找该日志器的上层日志器上所设置的日志处理器来处理日志记录。simpleExample1在配置文件中没有被定义,所以logging.getLogger(simpleExample1)
这行代码这是获取了一个logger实例,并无给它设置任何处理器,可是它的上级日志器--root logger在配置文件中有定义且设置了一个FileHandler处理器,simpleExample1处理器最终经过这个FileHandler处理器将日志记录输出到logging.log文件中了。
Python 3.2中引入的一种新的配置日志记录的方法--用字典来保存logging配置信息。这相对于上面所讲的基于配置文件来保存logging配置信息的方式来讲,功能更增强大,也更加灵活,由于咱们可把不少的数据转换成字典。好比,咱们可使用JSON格式的配置文件、YAML格式的配置文件,而后将它们填充到一个配置字典中;或者,咱们也能够用Python代码构建这个配置字典,或者经过socket接收pickled序列化后的配置信息。总之,你可使用你的应用程序能够操做的任何方法来构建这个配置字典。
这个例子中,咱们将使用YAML格式来完成与上面一样的日志配置。
首先须要安装PyYAML模块:
pip install PyYAML
Python代码:
import logging import logging.config import yaml with open('logging.yml', 'r') as f_conf: dict_conf = yaml.load(f_conf) logging.config.dictConfig(dict_conf) logger = logging.getLogger('simpleExample') logger.debug('debug message') logger.info('info message') logger.warn('warn message') logger.error('error message') logger.critical('critical message')
logging.yml配置文件的内容:
version: 1 formatters: simple: format: '%(asctime)s - %(name)s - %(levelname)s - %(message)s' handlers: console: class: logging.StreamHandler level: DEBUG formatter: simple stream: ext://sys.stdout console_err: class: logging.StreamHandler level: ERROR formatter: simple stream: ext://sys.stderr loggers: simpleExample: level: DEBUG handlers: [console] propagate: yes root: level: DEBUG handlers: [console_err]
输出结果:
2017-05-21 14:19:31,089 - simpleExample - DEBUG - debug message 2017-05-21 14:19:31,089 - simpleExample - INFO - info message 2017-05-21 14:19:31,089 - simpleExample - WARNING - warn message 2017-05-21 14:19:31,089 - simpleExample - ERROR - error message 2017-05-21 14:19:31,090 - simpleExample - CRITICAL - critical message
dictConfig()
函数的说明:该函数其实是对configparser
模块的封装,关于configparser
模块的介绍请参考<
该函数定义在loging.config模块下:
logging.config.dictConfig(config)
该函数能够从一个字典对象中获取日志配置信息,config参数就是这个字典对象。关于这个字典对象的内容规则会在下面进行描述。
不管是上面提到的配置文件,仍是这里的配置字典,它们都要描述出日志配置所须要建立的各类对象以及这些对象之间的关联关系。好比,能够先建立一个名额为“simple”的格式器formatter;而后建立一个名为“console”的处理器handler,并指定该handler输出日志所使用的格式器为"simple";而后再建立一个日志器logger,并指定它所使用的处理器为"console"。
传递给dictConfig()函数的字典对象只能包含下面这些keys,其中version
是必须指定的key,其它key都是可选项:
key名称 | 描述 |
---|---|
version | 必选项,其值是一个整数值,表示配置格式的版本,当前惟一可用的值就是1 |
formatters | 可选项,其值是一个字典对象,该字典对象每一个元素的key为要定义的格式器名称,value为格式器的配置信息组成的dict,如format和datefmt |
filters | 可选项,其值是一个字典对象,该字典对象每一个元素的key为要定义的过滤器名称,value为过滤器的配置信息组成的dict,如name |
handlers | 可选项,其值是一个字典对象,该字典对象每一个元素的key为要定义的处理器名称,value为处理器的配置信息组成的dcit,如class、level、formatter和filters,其中class为必选项,其它为可选项;其余配置信息将会传递给class所指定的处理器类的构造函数,以下面的handlers定义示例中的stream、filename、maxBytes和backupCount等 |
loggers | 可选项,其值是一个字典对象,该字典对象每一个元素的key为要定义的日志器名称,value为日志器的配置信息组成的dcit,如level、handlers、filters 和 propagate(yes |
root | 可选项,这是root logger的配置信息,其值也是一个字典对象。除非在定义其它logger时明确指定propagate值为no,不然root logger定义的handlers都会被做用到其它logger上 |
incremental | 可选项,默认值为False。该选项的意义在于,若是这里定义的对象已经存在,那么这里对这些对象的定义是否应用到已存在的对象上。值为False表示,已存在的对象将会被从新定义。 |
disable_existing_loggers | 可选项,默认值为True。该选项用于指定是否禁用已存在的日志器loggers,若是incremental的值为True则该选项将会被忽略 |
handlers定义示例:
handlers: console: class : logging.StreamHandler formatter: brief level : INFO filters: [allow_foo] stream : ext://sys.stdout file: class : logging.handlers.RotatingFileHandler formatter: precise filename: logconfig.log maxBytes: 1024 backupCount: 3
须要说明的是,上面所使用的对象并不限于loggging模块所提供的对象,咱们能够实现本身的formatter或handler类。另外,这些类的参数也许须要包含sys.stderr这样的外部对象。若是配置字典对象是使用Python代码构造的,能够直接使用sys.stdout、sys.stderr;可是当经过文本文件(如JSON、YAML格式的配置文件)提供配置时就会出现问题,由于在文本文件中,没有标准的方法来区分sys.stderr
和字符串'sys.stderr'
。为了区分它们,配置系统会在字符串值中查找特定的前缀,例如'ext://sys.stderr'中'ext://'会被移除,而后import sys.stderr
。