新闻中心
解决Python logging 中 datefmt 导致时间戳固定不变的问题

本文探讨了在python `logging` 配置中,当 `datefmt` 参数被错误地设置为 `datetime.now()` 的结果时,导致所有日志记录显示相同时间戳的问题。通过深入分析 `%(asctime)s` 和 `%(msecs)03d` 的工作机制,文章提供了两种解决方案:一是移除 `datefmt` 以利用默认的时间戳格式,二是通过在 `format` 字符串中显式添加 `%(msecs)03d` 来实现带有毫秒的自定义时间格式,确保日志时间戳的动态性和准确性。
在Python应用程序中,日志记录是调试、监控和故障排除的关键组成部分。logging 模块提供了高度灵活的配置选项,允许开发者自定义日志的输出格式、目的地和级别。然而,在配置日志时间戳时,一个常见的误区可能导致所有日志记录的时间戳显示为同一个值,尤其是在尝试使用 datefmt 参数来自定义时间格式时。本文将深入分析这一问题的原因,并提供两种有效的解决方案。
理解 logging 中的时间戳字段
在 logging.Formatter 中,有两个主要的格式化字段与时间戳相关:
-
%(asctime)s: 这是日志记录时间的主要字段。它的具体格式由 Formatter 实例的 datefmt 参数控制。
- 如果 datefmt 未指定,%(asctime)s 将使用默认格式,该格式通常包含毫秒(例如 YYYY-MM-DD HH:MM:SS,mmm)。
- 如果 datefmt 已指定,%(asctime)s 将严格按照 datefmt 字符串进行格式化。这意味着,如果 datefmt 只定义到秒,那么 %(asctime)s 也只会显示到秒,而不会包含毫秒。
- %(msecs)03d: 这个字段专门用于显示当前日志记录的毫秒部分。它总是输出三位数字,不足三位时前导补零。这个字段与 datefmt 参数无关,它直接从日志记录的时间戳中提取毫秒信息。
问题根源:datefmt 中的一次性求值
当在 logging 配置字典中将 datefmt 参数设置为 datetime.now().isoformat(...) 或类似的时间函数调用时,问题便会浮现。例如:
'datefmt': datetime.now().isoformat(sep='T',timespec='milliseconds')
这里的关键在于 datetime.now().isoformat(...) 表达式会在 logging_config 字典被创建(即脚本加载 logger_settings.py 文件时)的那一刻被执行一次。这意味着 datefmt 的值是一个固定的字符串,它代表了配置加载时的那个特定时间点。
因此,所有后续的日志记录,无论它们何时发生,其 %(asctime)s 字段都会被这个固定的 datefmt 值所格式化。如果 datefmt 字符串本身不包含动态的毫秒占位符,那么所有日志条目的时间戳看起来就如同“静止”了一般,无法反映出不同日志事件之间微小的时间差异。
解决方案
为了确保日志时间戳的动态性和毫秒精度,我们可以采用以下两种解决方案:
方案一:利用 %(asctime)s 的默认毫秒精度
如果对时间戳的格式要求不高,并且希望 %(asctime)s 自动包含毫秒,最简单的方法是不设置 datefmt 参数,或者将其设置为 None。
ChatGPT Writer
免费 Chrome 扩展程序,使用 ChatGPT AI 生成电子邮件和消息。
106
查看详情
示例配置:
import logging
import sys
from datetime import datetime
import time
logging_config_default_msecs = dict(
version=1,
formatters={
'verbose': {
'format': ("%(asctime)s %(levelname)s "
"[%(name)s:%(lineno)s] %(message)s"),
# 'datefmt': None # 或者直接移除此行,%(asctime)s将使用默认格式
},
'simple': {
'format': '%(asctime)s %(levelname)-8s %(message)s',
# 'datefmt': None
},
},
handlers={
'console': {
'class': 'logging.StreamHandler',
'level': 'DEBUG',
'formatter': 'verbose',
'stream': sys.stdout,
},
},
loggers={
'app_logger': {
'handlers': ['console'],
'level': logging.DEBUG,
'propagate': False
},
}
)
# 示例使用
if __name__ == '__main__':
from logging.config import dictConfig
print("--- 方案一:利用 %(asctime)s 默认毫秒精度 ---")
dictConfig(logging_config_default_msecs)
logger = logging.getLogger('app_logger')
logger.info("这是第一条日志 (方案一)")
time.sleep(0.005) # 模拟时间间隔
logger.info("这是第二条日志 (方案一)")
time.sleep(0.008)
logger.info("这是第三条日志 (方案一)")在这种配置下,%(asctime)s 将使用 logging 模块的默认格式,通常为 YYYY-MM-DD HH:MM:SS,mmm,自动包含毫秒。
方案二:自定义 datefmt 并显式添加 %(msecs)03d (推荐)
如果需要自定义 datefmt 的格式(例如 ISO 8601 格式,但不希望 datefmt 自身处理毫秒),可以将 datefmt 设置为只包含年月日时分秒的格式,然后在 format 字符串中显式地添加 %(msecs)03d 来显示毫秒。这是最灵活且符合期望的解决方案,它允许你精确控制日期时间的整体格式,同时确保毫秒部分的动态性。
示例配置:
import logging
import sys
from datetime import datetime
import time
logging_config_custom_msecs = dict(
version=1,
formatters={
'verbose': {
'format': ("%(asctime)s.%(msecs)03d %(levelname)s " # 显式添加 .%(msecs)03d
"[%(name)s:%(lineno)s] %(message)s"),
'datefmt': '%Y-%m-%dT%H:%M:%S' # datefmt 只定义到秒
},
'simple': {
'format': '%(asctime)s.%(msecs)03d %(levelname)-8s %(message)s',
'datefmt': '%Y-%m-%dT%H:%M:%S'
},
},
handlers={
'file_handler': {'class': 'logging.FileHandler',
'formatter': 'verbose',
'level': logging.DEBUG,
'filename': '/tmp/logtest_'+datetime.now().strftime("%Y%m%d-%H%M%S")+'.log'},
'console': {
'class': 'logging.StreamHandler',
'level': 'DEBUG',
'formatter': 'simple',
'stream': sys.stdout,
},
},
loggers={
'app_logger': {
'handlers': ['file_handler', 'console'], # 确保引用正确的handler名称
'level': logging.DEBUG,
'propagate': False
},
}
)
# 示例使用
if __name__ == '__main__':
from logging.config import dictConfig
print("\n--- 方案二:自定义 datefmt 并显式添加 %(msecs)03d ---")
dictConfig(logging_config_custom_msecs)
logger = logging.getLogger('app_logger')
logger.info("这是第一条日志 (方案二)")
time.sleep(0.005) # 模拟时间间隔
logger.info("这是第二条日志 (方案二)")
time.sleep(0.008)
logger.info("这是第三条日志 (方案二)")在这个方案中,%(asctime)s 会按照 '%Y-%m-%dT%H:%M:%S' 格式化(只到秒),然后 .%(msecs)03d 会追加当前日志记录的毫秒数。这样既实现了自定义的日期时间格式,又确保了毫秒的动态显示。
注意事项与最佳实践
- 避免在 logging 配置字典的 datefmt 值中直接调用 datetime.now() 或其他时间函数,除非你确实希望其值在配置加载时被固定。对于需要动态更新的值,应使用 Formatter 提供的占位符。
- 理解 %(asctime)s 和 %(msecs)03d 的区别和作用。%(asctime)s 是一个复合字段,其格式受 datefmt 影响;而 %(msecs)03d 是一个独立的、始终表示毫秒的字段。
- 对于文件名的动态生成,例如 filename': '/tmp/logtest_'+datetime.now().strftime("%Y%m%d-%H%M%S")+'.log',使用 datetime.now().strftime(...) 是完全正确的。因为文件名只需要在应用启动时确定一次,并且它不会影响日志记录内部的时间戳格式。
通过采用上述解决方案,可以有效避免Python logging 中 datefmt 配置不当导致的时间戳固定问题,确保日志的准确性和实用性。
以上就是解决Python logging 中 datefmt 导致时间戳固定不变的问题的详细内容,更多请关注其它相关文章!
# 数据处理
# yas4444seo
# 英山seo服务好
# 车后面字母seo
# bing搜索的seo
# 福田各大网站推广优化
# 营销推广入门试题
# 官方推广营销方案
# 泰康医疗推广官方网站
# 广州大学网站建设
# 封丘seo推广
# 第三条
# 第二条
# 第一条
# python
# 加载
# 两种
# 设置为
# 是一个
# 自定义
# 这是
# gate
# yy
# 区别
# stream
# ai
# app
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
期待已久:小米17 Ultra、小米首款NAS本月登场
三星ZFold5多任务卡顿_Samsung ZFold5流畅度提升
Golang如何优化内存分配与垃圾回收_Golang内存管理与GC优化实践
C++的std::mdspan是什么_C++23中用于操作多维数组的非拥有视图
如何使 Jest 模拟函数默认抛出错误以提高测试效率
HuggingFaceEmbeddings中向量嵌入维度调整的限制与理解
在VS Code中配置和运行Dart程序的完整步骤
今日头条怎么同步内容到抖音_今日头条内容同步到抖音教程
CSS布局:解决全屏元素100%尺寸与外边距导致的页面溢出问题
MAC的“快捷指令”怎么同步到iPhone_MAC利用iCloud同步所有设备的自动化指令
J*aScript map 方法中处理循环元素为空数组的策略
聚水潭ERP登录页面入口 聚水潭ERP官网登录界面
神经网络二分类模型训练异常:高损失与完美验证准确率的排查与修正
Typer应用中动态命令行参数的解析与处理
夸克浏览器图书入口 夸克手机浏览器阅读入口
在Pyomo中实现基于变量的条件约束:Big-M方法详解
汽水音乐网页版使用入口_汽水音乐电脑版播放指南
Python异步编程实践:使用Binance API构建实时交易数据流
豆包手机助手发布技术预览版:直接嵌入手机系统!努比亚样机发售
解决 MongoDB 聚合查询中对象数组 _id 匹配问题
Go语言中Map值调用指针接收器方法的限制与应对
Selenium Python中处理点击后新窗口加载冻结问题的策略与实践
ArrayList与LinkedList操作复杂度详解:遍历与修改
汽水音乐在线版入口_汽水音乐网页播放手册
J*aScript对象创建方式_J*aScript设计模式应用
俄罗斯方块最新版入口 俄罗斯方块在线玩官网入口
Linux如何构建多环境配置管理_Linux多环境配置方案
冬*霸灯泡不亮怎么办_浴霸取暖灯一盏不亮的灯座清洁修复法
React列表渲染与独立状态管理:避免全局状态影响局部更新
可靠CSGO开箱平台解析 CSGO开箱网合集
Lar*el 递归关系中排除指定分支的教程
小米14应用无法联网原因分析_小米14网络权限修复
一加Ace 6T支持全新明眸护眼:通过了最严苛的护眼小金标认证
如何将HTML表格多行数据保存到Google Sheet
Composer的 "check-platform-reqs" 命令有什么用_在部署前检查生产环境是否满足Composer依赖需求
京东单号查询入口_京东快递订单追踪入口
抓大鹅解压小游戏 抓大鹅摸鱼解压入口
微博网页版首页入口 微博电脑端官网登录链接
如何在Python中使用Optional类型处理可变对象并避免Pylint警告
win11如何卸载Windows更新补丁 Win11解决更新导致系统不稳定的问题【修复】
谷歌浏览器一键优化方案_谷歌浏览器直达主页极速不卡版
如何在离线环境中使用Composer_Composer离线安装依赖包的技巧与策略
支付宝解绑银行卡步骤_支付宝如何解除绑定银行卡
现代化 SciPy 一维插值:interp1d 的替代方案与最佳实践
谷歌学术网站直达地址 谷歌学术搜索网页版一键进入
UC浏览器如何安装插件 UC浏览器添加扩展程序详细教程【进阶】
在Qt QML中通过Python字典动态更新TextEdit内容的教程
css元素hover动画延迟生效怎么办_使用animation-delay调整触发时间
提升屏幕阅读器对“m”时间单位的播报准确性:HTML与CSS组合解决方案
移动端XML文件怎么转换成Excel 手机和平板上的解决方案


2025-12-01
浏览次数:次
返回列表