Python学习打call第四十一天:logging日志模块

1.什么是日志级别

  • 日志级别是日志的严重程度,如果设置的日志级别低于logging模块设置的级别,那么将被忽略;

  • warning是默认的日志级别,criticalerror的级别高于warning,而infodebug级别高于warning

    2d143e117de30d40.png

2.日志级别实例

我们知道默认的日志级别是warning,那下面我们通过一个实例来对比一下warning日志级别和其他的日志级别的等级高级,看看如果设置的日志级别低于logging模块设置的级别,那么将被忽略这个结论是否正确

import logging
from datetime import datetime

logging.basicConfig(format='%(message)s {}'.format(datetime.now()), level=logging.WARNING)

logging.debug('我是 DEBUG 级别的日志')
logging.info('我是 INFO 级别的日志')
logging.warning('我是 WARNING 级别的日志')
logging.error('我是 ERROR 级别的日志')
logging.critical('我是 CRITICAL 级别的日志')


输出结果:
我是 WARNING 级别的日志 2019-04-11 15:37:14.254115
我是 ERROR 级别的日志 2019-04-11 15:37:14.254115
我是 CRITICAL 级别的日志 2019-04-11 15:37:14.254115

通过输出结果,我们知道了上面的那个结论是正确的啦~

我们再用其他的日志级别实践一下,日志级别为DEBUG,那只要日志级别比DEBUG高的都会打印;

import logging
from datetime import datetime

logging.basicConfig(format='%(message)s {}'.format(datetime.now()), level=logging.DEBUG)

logging.debug('我是 DEBUG 级别的日志')
logging.info('我是 INFO 级别的日志')
logging.warning('我是 WARNING 级别的日志')
logging.error('我是 ERROR 级别的日志')
logging.critical('我是 CRITICAL 级别的日志')

输出结果:

我是 DEBUG 级别的日志 2019-04-11 15:39:25.098599
我是 INFO 级别的日志 2019-04-11 15:39:25.098599
我是 WARNING 级别的日志 2019-04-11 15:39:25.098599
我是 ERROR 级别的日志 2019-04-11 15:39:25.098599
我是 CRITICAL 级别的日志 2019-04-11 15:39:25.098599

3.日志格式参数有哪些

  • %(levelno)s:打印日志级别的数值;

  • %(levelname)s:打印日志级别的名称;

  • %(pathname)s:打印当前执行程序的路径,其实就是sys.argv[0];

  • %(filename)s:打印当前执行程序名;

  • %(funcName)s:打印日志的当前函数;

  • %(lineno)d:打印日志的当前行号;

  • %(asctime)s:打印日志的时间;

  • %(thread)d:打印线程ID;

  • %(threadName)s:打印线程名称;

  • %(process)d:打印进程ID;

  • %(processName)s:打印线程名称;

  • %(module)s:打印模块名称;

  • %(message)s:打印日志信息;

4.日志时间格式化

打印日志的时候,时间精确到了毫秒,但有时候我们不需要精确到毫秒,那要怎么办?可以使用日志时间格式化,也就是设置datefmt

import logging
from datetime import datetime

logging.basicConfig(format='%(asctime)s  %(message)s {}'.format(datetime.now()), level=logging.WARNING, datefmt='%Y-%m-%d')

logging.debug('我是 DEBUG 级别的日志')
logging.info('我是 INFO 级别的日志')
logging.warning('我是 WARNING 级别的日志')
logging.error('我是 ERROR 级别的日志')
logging.critical('我是 CRITICAL 级别的日志')

输出结果:
2019-04-11  我是 WARNING 级别的日志 2019-04-11 15:43:41.166245
2019-04-11  我是 ERROR 级别的日志 2019-04-11 15:43:41.166245
2019-04-11  我是 CRITICAL 级别的日志 2019-04-11 15:43:41.166245

通过上面的例子,我们知道可以使用datefmt来设置打印时间的格式,如打印年月日时分秒为:datefmt='%Y-%m-%d %H:%M:%S';

5.日志输出处理 Handler

import logging
from pprint import pprint

logging.basicConfig(format='%(asctime)s  %(message)s ', level=logging.DEBUG, datefmt='%Y-%m-%d %H:%M:%S')
# 通过logging的工厂方法,返回logger实例
root_logger = logging.getLogger()
xkd_logger = logging.getLogger('xkd')
root_logger.setLevel(40)

# handler的level级别必须大于等于 logger实例的level级别才有意义
# 如果日志的级别低于logger的级别是无法进入到logger的
error_handler = logging.StreamHandler()
error_handler.setLevel(level=logging.ERROR)
error_handler.setFormatter(fmt=logging.Formatter('%(asctime)s  %(message)s', datefmt='%Y-%m-%d %H:%M:%S'))

critical_handler = logging.FileHandler(filename='./app.log')
critical_handler.setLevel(level=logging.CRITICAL)
critical_handler.setFormatter(fmt=logging.Formatter('%(asctime)s  %(message)s', datefmt='%Y-%m-%d %H:%M:%S'))

root_logger.addHandler(error_handler)
root_logger.addHandler(critical_handler)
root_logger.info('我是 root 的 INFO 级别的日志')
root_logger.error('我是 root 的 ERROR 级别的日志')
root_logger.critical('我是 root 的 CRITICAL 级别的日志')
pprint(root_logger.handlers)

# 子logger
xkd_level = xkd_logger.getEffectiveLevel()
print(xkd_level)

xkd_logger.info('我是 INFO 级别')  # 不会打印
xkd_logger.error('我是 error 级别')  # 会打印
  • handler的level级别必须大于等于logger实例的level级别才有意义,如果日志的级别低于logger的级别是无法进入到logger的;

  • 日志的输出可以是控制台,可以是文件, 默认是终端 stderr;

  • logger实例有继承关系,如果子logger没有设置level和handler,那么从父logger中继承;

  • 每一个logger实例,默认有一个handler,StreamHandler;

  • 每一个logger实例都有一个propagate属性,控制日志信息是否向父类传递, 默认propagate为True,则向父类传递日志信息;

6.第三方日志模块structlog的使用

安装命令行:

pip install structlog

# 或者带颜色打印(如果是打印到文件,最好不要使用带颜色打印,会有转义字符)
pip install structlog colorama

我们来看一下structlog的基本使用

from structlog import get_logger
log = get_logger()
# 可以直接传递key/value
log.info("需要打印的信息", out_of_the_box=True, effort=0)

# 输出结果
# 2019-04-11 15:53.19 需要打印的信息   effort=0 out_of_the_box=True

首先需要导入structlog模块,然后import get_logger类,接着使用get_logger进行实例化,得到实例对象log,最后就可以直接使用实例对象log进行打印了;

7.自定义全局日志模块

使用标准库logging模块封装全局日志

import logging
import getpass
import sys
import os

#定义Mylog类,管理log信息
class MyLog(object):
    def __init__(self):
        self.user = getpass.getuser()
        self.logger = logging.getLogger(self.user)
        self.logger.setLevel(logging.DEBUG)
        #日志文件名
        self.logFile = 'debug.log'
        self.formatter = logging.Formatter('%(asctime)-12s %(levelname)- 8s %(name)-10s %(message)-12s\r\n')
        #日志显示到屏幕并输出到文档
        self.logHand = logging.FileHandler(self.logFile, encoding='utf-8')
        self.logHand.setFormatter(self.formatter)
        self.logHand.setLevel(logging.DEBUG)
        self.logHandSt = logging.StreamHandler()
        self.logHandSt.setFormatter(self.formatter)
        self.logHandSt.setLevel(logging.DEBUG)
        self.logger.addHandler(self.logHand)
        self.logger.addHandler(self.logHandSt)
    #日志的5个级别对应5个函数
    def debug(self, msg):
        self.logger.debug(msg)
    def info(self, msg):
        self.logger.info(msg)
    def warn(self, msg):
        self.logger.warning(msg)
    def error(self, msg):
        self.logger.error(msg)
    def critical(self, msg):
        self.logger.critical(msg)

#测试代码 __name__是访问当前函数名的方法,如果作为模块就不是main函数 下面的方法不会执行
if __name__ == '__main__':
    mylog = MyLog()
    mylog.debug(u"I'm 测试中文")
    mylog.info("I;m info")
    mylog.warn("warning")
    mylog.error("Error")
    mylog.critical("this is a critical")
    

# 输出:
2019-04-11 16:04:50,836 DEBUG    lu         I'm 测试中文    
2019-04-11 16:04:50,836 INFO     lu         I;m info    
2019-04-11 16:04:50,837 WARNING  lu         warning     
2019-04-11 16:04:50,837 ERROR    lu         Error       
2019-04-11 16:04:50,837 CRITICAL lu         this is a critical
  • 使用标准库logging模块封装全局日志;

  • 首先需要构造一个类,然后初始化这个类,设置对应五个日志级别的五个函数,最后实例化这个类,就可以通过实例对象直接打印了;

参考:https://www.9xkd.com/user/plan-view.html?id=4239617430

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,591评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,448评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,823评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,204评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,228评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,190评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,078评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,923评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,334评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,550评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,727评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,428评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,022评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,672评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,826评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,734评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,619评论 2 354

推荐阅读更多精彩内容

  • 本文章是我大概三年前,在上家单位使用 Python 工作时结合官方文档做的整理。现在 Python 官方文档听说已...
    好吃的野菜阅读 216,869评论 14 232
  • 本文翻译自logging howto 基础教程 日志是跟踪软件运行时发生事件的一种手段。Python开发者在代码中...
    大蟒传奇阅读 4,253评论 0 17
  • 在现实生活中,记录日志非常重要。银行转账时会有转账记录;飞机飞行过程中,会有黑盒子(飞行数据记录器)记录飞行过程中...
    chliar阅读 765评论 1 0
  • 谷歌浏览器一款非常好用的调试工具,可以模拟浏览器和客户端。例如,支付中用到支付宝,点击支付,可以暂停支付,然后查看...
    奶茶007阅读 1,734评论 0 0
  • 仉咏阅读 158评论 0 6