Python日志模块:实战应用与最佳实践

本文详细解析了Python的logging模块,从基本介绍到实际应用和最佳实践。我们通过具体的代码示例解释了如何高效地使用这个模块进行日志记录,以及如何避免常见的陷阱,旨在帮助读者更好地掌握这个强大的工具。
一、Python 日志模块简介
日志的概念及其在软件开发中的作用
在开发过程中,为了记录应用程序的运行情况,通常我们会采用打印日志的方式,这种方式不仅可以帮助我们了解软件系统的运行状态,还可以在系统出现错误时,帮助我们快速定位问题。
例如,假设你有以下一段代码,它只是简单地输出一些信息:
print("This is some information.")
输出:
This is some information.
但是,如果我们需要记录更复杂的信息,如错误信息、警告或者其他重要的运行时信息,仅仅使用print就显得力不从心。这就是我们需要日志模块的地方。
Python logging日志模块简介
Python内置的logging模块为我们提供了一套完整的日志记录解决方案。在许多情况下,你可能希望你的应用程序能够在运行时输出某种形式的状态信息,特别是当应用程序需要处理长时间运行的任务,或者当你面临需要诊断的问题时,logging模块便是你的得力助手。
logging模块可以帮助我们捕获、处理和记录日志信息,使得我们可以在程序运行的任何地方快速记录日志信息。相比于简单的print函数,它更具有灵活性,能够同时输出日志到多个地方,如:控制台、文件、HTTP GET/POST,SMTP,Socket等,并且可以独立设置每个输出的日志等级。
以下是一个简单的例子来说明如何使用logging模块:
import logging
# Create a logger and set the log level to INFO
logger = logging.getLogger(__name__)
logger.setLevel(logging.INFO)
# Add a StreamHandler to send log messages to console
console_handler = logging.StreamHandler()
logger.addHandler(console_handler)
# Log an informational message
logger.info("This is an informational message.")
这段代码会输出以下信息到控制台:
This is an informational message.
logging模块的基本组成
logging模块主要由以下几个部分组成:
Logger: 用于提供应用程序直接使用的接口。
Handler: 将(logger产生的)日志记录发送到合适的目的输出。
Filter: 提供了更精细的工具来决定输出哪些日志记录。
Formatter: 指定日志记录的最终输出格式。
二、logging日志模块详解
logging的基础使用
使用Python的logging模块相当简单,下面是一个基本的例子,说明如何创建一个日志并输出到控制台。
import logging
# This will log the message to the console
logging.warning('This is a warning message')
这段代码将输出以下警告信息:
WARNING:root:This is a warning message
理解日志级别
在logging模块中,我们有5个级别来描述日志的重要性。这些级别分别是:
DEBUG:详细信息,通常仅在诊断问题时使用。
INFO:确认事情按预期进行。
WARNING:出现了一些预期之外的事情,或者在不久的将来可能出现问题(例如,“磁盘空间不足”)。但是软件仍在正常工作。
ERROR:由于更严重的问题,软件不能执行某些功能。
CRITICAL:严重的错误,表明程序本身可能无法继续运行。
默认情况下,logging模块将日志记录到控制台,并且只处理级别为WARNING以上的日志。
Loggers、Handlers和Formatters
这一部分我们将会详解Loggers、Handlers和Formatters这三个主要组件。
Loggers的作用和使用
Logger是一个日志对象,主要任务就是记录日志。在应用程序代码中任何需要日志的地方,都可以创建一个logger实例,并用其记录需要的信息。下面是一个简单的使用logger的例子:
import logging
# Create a logger
logger = logging.getLogger(__name__)
# Log some messages
logger.debug("This is a debug message.")
logger.info("This is an informational message.")
logger.warning("Careful! Something does not look right.")
logger.error("You have encountered an error.")
logger.critical("The program cannot recover from this situation!")
注意:当我们运行这段代码时,我们并没有看到任何输出。这是因为默认情况下,logger的级别设置为WARNING,因此只有级别为WARNING以上的日志会被处理。
Handlers的种类和作用
Handler对象负责发送日志记录到合适的目的地。不同的handler可以将日志发送到控制台,文件,邮件,甚至HTTP POST参数等。下面是一个简单的例子,说明如何使用handler将日志记录到文件和控制台:
import logging
# Create a logger
logger = logging.getLogger(__name__)
logger.setLevel(logging.DEBUG)
# Create a file handler
file_handler = logging.FileHandler('my_log.log')
logger.addHandler(file_handler)
# Create a console handler
console_handler = logging.StreamHandler()
logger.addHandler(console_handler)
# Log some messages
logger.debug("This is a debug message.")
logger.info("This is an informational message.")
logger.warning("Careful! Something does not look right.")
logger.error("You have encountered an error.")
logger.critical("The program cannot recover from this situation!")
Formatters的功能和自定义日志格式
Formatter对象指定日志记录的最终顺序,结构和内容。你可以自定义日志信息的格式,使得日志信息更具有可读性。下面是一个如何使用formatter的例子:
import logging
# Create a logger
logger = logging.getLogger(__name__)
logger.setLevel(logging.DEBUG)
# Create a console handler
console_handler = logging.StreamHandler()
# Create a formatter
formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
# Add the formatter to the console handler
console_handler.setFormatter(formatter)
# Add the console handler to the logger
logger.addHandler(console_handler)
# Log some messages
logger.debug("This is a debug message.")
logger.info("This is an informational message.")
logger.warning("Careful! Something does not look right.")
logger.error("You have encountered an error.")
logger.critical("The program cannot recover from this situation!")
三、Python日志模块在实践中的应用
使用日志记录异常信息
在Python编程中,经常需要捕获和处理异常。这时,使用logging模块记录异常信息会非常方便。在logging模块中,我们可以使用exception()方法记录异常堆栈信息。如下例所示:
import logging
logger = logging.getLogger(__name__)
try:
a = [1, 2, 3]
value = a[3]
except IndexError as e:
logger.error("Unhandled exception", exc_info=True)
```
当运行此段代码,日志记录器将记录下出现的异常信息,如下:
```python
ERROR:__main__:Unhandled exception
Traceback (most recent call last):
File "<stdin>", line 4, in <module>
IndexError: list index out of range
使用RotatingFileHandler进行日志滚动
当我们的应用程序运行很长时间,并产生大量的日志时,所有的日志都写入一个文件可能会导致这个日志文件过大。这时,我们可以使用RotatingFileHandler来进行日志滚动。当达到一定的大小或者一定的时间,RotatingFileHandler会自动备份当前日志文件,并创建一个新的日志文件继续写入。如下例所示:
import logging
from logging.handlers import RotatingFileHandler
logger = logging.getLogger(__name__)
logger.setLevel(logging.INFO)
# Create a file handler
handler = RotatingFileHandler('my_log.log', maxBytes=2000, backupCount=10)
logger.addHandler(handler)
# Log some messages
for _ in range(10000):
logger.info("Hello, world!")
```
这段代码将在日志文件大小达到2000字节时创建一个新的日志文件,并保留最新的10个日志文件。
## 配置日志级别
根据我们的需要,可以在运行时改变日志的级别。例如,当我们在调试应用程序时,我们可能需要输出所有级别的日志。但是在生产环境中,我们可能只关心错误及以上级别的日志。我们可以通过setLevel()函数来改变日志级别。如下例所示:
```python
import logging
# Create a logger
logger = logging.getLogger(__name__)
# Set log level to DEBUG
logger.setLevel(logging.DEBUG)
# Log some messages
logger.debug("This is a debug message.")
logger.info("This is an informational message.")
logger.warning("Careful! Something does not look right.")
logger.error("You have encountered an error.")
logger.critical("The program cannot recover from this situation!")
四、Python日志模块的最佳实践
在模块级别使用__name__创建logger
在Python中,__name__变量是一个内置变量,它代表当前模块的名称。当我们在每个模块级别上创建logger并使用__name__作为名称,我们可以轻松地追踪日志记录发生在哪个模块。
import logging
# Create a logger at the module level
logger = logging.getLogger(__name__)
使用合适的日志级别
不同的日志级别表示了不同的严重性。正确地使用日志级别可以帮助我们在大量的日志中找到我们关心的信息。一般来说,对于非常严重的错误,我们应使用CRITICAL或ERROR;对于警告信息,我们应使用WARNING;对于常规的运行信息,我们应使用INFO;对于调试信息,我们应使用DEBUG。
使用结构化的日志消息
当我们的应用程序有大量的日志时,我们可能希望以一种可解析的方式记录日志消息。例如,我们可以使用JSON格式记录日志。这样,我们就可以使用各种日志分析工具分析日志。
import logging
import json
# Create a logger
logger = logging.getLogger(__name__)
# Log a structured message
logger.info(json.dumps({
'action': 'User login',
'username': 'user123',
'ip_address': '123.123.123.123',
'status': 'success',
}))
使用异常日志记录
当捕获到异常时,我们应使用logger.exception(),这样就可以在日志中记录下完整的异常堆栈信息。
import logging
logger = logging.getLogger(__name__)
try:
x = 1 / 0
except ZeroDivisionError:
logger.exception("Zero Division Error Caught.")
```
这样的日志会包含足够的信息帮助我们找到和修复问题。
## 不要在日志中记录敏感信息
日志可能被攻击者用来寻找系统的漏洞,因此我们绝对不能在日志中记录敏感信息,如密码、密钥和用户的私有数据。
# 五、总结
在这篇文章中,我们详细介绍了Python的logging模块,包括它的基本介绍,详解,实践中的应用,以及一些最佳实践。总结上述内容:
1. logging模块是Python内置的一种灵活且强大的日志记录工具,它可以将程序运行过程中的信息输出到各种输出源,如标准输出、文件、邮件、网络等。
2. logging模块提供了多种级别的日志记录,包括DEBUG,INFO,WARNING,ERROR和CRITICAL。我们可以根据需求设置不同的日志级别,以记录和展示不同严重性的信息。
3. 在实践中,我们可以使用logging模块来记录异常信息,使用RotatingFileHandler进行日志滚动,以及在运行时改变日志级别。
4. 对于logging模块的最佳实践,我们提到了在模块级别使用__name__创建logger,使用合适的日志级别,使用结构化的日志消息,使用异常日志记录,以及不在日志中记录敏感信息。
**Python的logging模块是一个非常强大的工具,希望你在阅读本文后能有更深的理解和更灵活的运用。**
> 如有帮助,请多关注
> 个人微信公众号:【Python全视角】
> TeahLead_KrisChang,10+年的互联网和人工智能从业经验,10年+技术和业务团队管理经验,同济软件工程本科,复旦工程管理硕士,阿里云认证云服务资深架构师,上亿营收AI产品业务负责人。
Python日志模块:实战应用与最佳实践的更多相关文章
- python日志模块logging
python日志模块logging 1. 基础用法 python提供了一个标准的日志接口,就是logging模块.日志级别有DEBUG.INFO.WARNING.ERROR.CRITICAL五种( ...
- 一文搞懂 Python 的模块和包,在实战中的最佳实践
最近公司有个项目,我需要写个小爬虫,将爬取到的数据进行统计分析.首先确定用 Python 写,其次不想用 Scrapy,因为要爬取的数据量和频率都不高,没必要上爬虫框架.于是,就自己搭了一个项目,通过 ...
- Python自动化运维 技术与最佳实践PDF高清完整版免费下载|百度云盘|Python基础教程免费电子书
点击获取提取码:7bl4 一.内容简介 <python自动化运维:技术与最佳实践>一书在中国运维领域将有"划时代"的重要意义:一方面,这是国内第一本从纵.深和实践角度探 ...
- python日志模块的使用
学习一下python的日志模块logging,可以参考如下博客,写得很详细 https://www.cnblogs.com/yyds/p/6901864.html https://www.cnblog ...
- python日志模块
许多应用程序中都会有日志模块,用于记录系统在运行过程中的一些关键信息,以便于对系 统的运行状况进行跟踪.在.NET平台中,有非常著名的第三方开源日志组件log4net,c++中,有人们熟悉的log4c ...
- Python 日志模块实例
python 打印对象的所有属性值: def prn_obj(obj): print '\n'.join(['%s:%s' % item for item in obj.__dict__.it ...
- Python日志模块logging用法
1.日志级别 日志一共分成5个等级,从低到高分别是:DEBUG INFO WARNING ERROR CRITICAL. DEBUG:详细的信息,通常只出现在诊断问题上 INFO:确认一切按预期运行 ...
- python日志模块logging学习
介绍 Python本身带有logging模块,其默认支持直接输出到控制台(屏幕),或者通过配置输出到文件中.同时支持TCP.HTTP.GET/POST.SMTP.Socket等协议,将日志信息发送到网 ...
- python日志模块笔记
前言 在应用中记录日志是程序开发的重要一环,也是调试的重要工具.但却很容易让人忽略.之前用flask写的一个服务就因为没有处理好日志的问题导致线上的错误难以察觉,修复错误的定位也很困难.最近恰好有时间 ...
- Java日志体系(八)最佳实践
java常用日志框架关系 Log4j 2与Log4j 1发生了很大的变化,Log4j 2不兼容Log4j 1. Logback必须配合Slf4j使用.由于Logback和Slf4j是同一个作者,其兼容 ...
随机推荐
- TEC-6计算机组成原理实验(简图)
TEC-6计算机组成原理实验
- RabbitMQ详解(上)
一:MQ的相关概念 MQ(message queue),从字面意思上看,本质是个队列,FIFO 先入先出,只不过队列中存放的内容是message 而已,还是一种跨进程的通信机制,用于上下游传递消息.在 ...
- “结巴”中文分词:做最好的 Python 中文分词组件
jieba "结巴"中文分词:做最好的 Python 中文分词组件 "Jieba" (Chinese for "to stutter") C ...
- 开发 Diffusers 库的道德行为指南
我们正在努力让我们每次发布的库更加负责! 我们很荣幸宣布我们发布了 道德守则,并将作为一部分其放入 Diffusers 库的说明文档. 由于扩散模型在现实世界上的实际应用例子会对社会造成潜在的负面影 ...
- Vue2到Vue3的改变
一.Vue2->Vue3 如果有Vue2的基础,并在此基础上学习Vue3,并不需要把完整的官网看完,我们只需要关注一下新功能和非兼容的变化即可进行开发. 二.Vue3变化 统一元素上使用的v-i ...
- 文心一言 VS chatgpt (5)-- 算法导论2.2 3~4题
三.再次考虑线性查找问题(参见练习 2.1-3).假定要查找的元素等可能地为数组中的任意元素,平均需要检查输入序列的多少元素?最坏情况又如何呢?用0记号给出线性查找的平均情况和最坏情况运行时间.证明你 ...
- 2022-12-03:部门工资最高的员工。以下数据Max 和 Jim 在 IT 部门的工资都是最高的,Henry 在销售部的工资最高。sql语句如何写? 输出结果如下: department emp
2022-12-03:部门工资最高的员工.以下数据Max 和 Jim 在 IT 部门的工资都是最高的,Henry 在销售部的工资最高.sql语句如何写? 输出结果如下: department empl ...
- 【设计模式】使用 go 语言实现简单工厂模式
最近在看<大话设计模式>,这本书通过对话形式讲解设计模式的使用场景,有兴趣的可以去看一下. 第一篇讲的是简单工厂模式,要求输入两个数和运算符号,得到运行结果. 这个需求不难,难就难在类要怎 ...
- 用Linux命令操作mysql数据库
操作mysql数据库,相信大家最熟悉的应该是用navicat工具来新建数据库,建表,查询数据,查看表结构等. 但是如果数据库与本操作机器不在同一个局域网内,并且对方环境也不支持vpn的情况下,如何查询 ...
- itextpdf5.5.13给pdf添加图片水印、添加文字水印(平铺)、添加文字水印(单个)、添加页眉、页脚、页眉事件、添加图片
转载自简书用户:alex很累,感谢分享.原地址:https://www.jianshu.com/p/2b9c7a0300e4 一.相关工具类 1. Excel2Pdf.java (如代码不可用请查看原 ...