flask_sqlalchemy时间缓存导致datetime.now()时间不变问题

问题是这样的,项目在本地没什么问题,但是部署到服务器过一阵子发现,这个时间会在某一刻定死不变。

重启uwsgi后,发现第一条数据更新到了目前最新时间,过了一会儿再次发送也变了时间,但是再过几分钟再发就会变成和上次一样的时间。

我的基类模型是这样写的

python 复制代码
class BaseModel(db.Model):
    """ 基类模型 """
    __abstract__ = True
    id = db.Column(db.Integer, primary_key=True, comment='id主键')
    add_time = db.Column(db.DateTime, default=datetime.now(), comment='创建时间')
    upd_time = db.Column(db.DateTime, default=datetime.now(), onupdate=datetime.now(), comment='更新时间')

从发送到保存都检查了一遍,没有发现什么问题,且检查了debian的时区和date是没问题的,因为python的datetime就是基于系统时间而言去生成的。

解决方案是需要将时间函数改成函数引用的方式,将now的函数地址传递。这也每次sqlalchemy生成执行sql的时候,将会直接执行函数引用,将 datetime.now当作参数进行传递,并执行。

最终将 datetime.now()改成datetime.now搞定

python 复制代码
class BaseModel(db.Model):
    """ 基类模型 """
    __abstract__ = True
    id = db.Column(db.Integer, primary_key=True, comment='id主键')
    add_time = db.Column(db.DateTime, default=datetime.now, comment='创建时间')
    upd_time = db.Column(db.DateTime, default=datetime.now, onupdate=datetime.now, comment='更新时间')

源码分析:


Columndefault参数通常接收一个默认的固定值以及一个可调用对象

处理列的默认值的关键类,这里的 __new__ 构造方法是决定 ColumnDefault 类实例化行为关键代码。

  • 如果 arg 是一个可调用对象(callable),则创建 CallableColumnDefault 类的实例。
  • 如果 arg 是 ClauseElement 的实例,创建 ColumnElementColumnDefault 类的实例。
  • 如果 arg 是非 None 的其他类型,创建 ScalarElementColumnDefault 类的实例。

如果你传入的是 datetime.now则会走 CallableColumnDefault,否则 now()的话就会视为常量而走 ScalarElementColumnDefault

当你将 datetime.now 作为 default 参数传递时(不带括号),它被视为一个可调用对象,因此会创建 CallableColumnDefault 的实例。这意味着每次需要默认值时,datetime.now 都会被调用,生成一个新的当前时间戳。

而当你传递 datetime.now() 时(带括号),它立即执行并返回一个具体的 datetime 对象。然而,这种用法通常不是期望的,因为它不会在每次插入操作时提供一个新的时间,而是使用了定义时的时间。

在实际运行时,SQLAlchemy 会根据 ColumnDefault 实例的类型来确定如何处理默认值

可以得到结论是在SQLAlchemy中,关于时间创建时自动生成的逻辑,是得用 datetime.now作为可调用对象传递过去,而不是常量。但是关于我在重启服务后,并没有直接定死,而是发了两条可以获得两个时间节点的数据后,再定死的逻辑没有查到具体的位置。按照分析如果我最初传入的是 now()则在第一次是 16:01 后再次发也是 16:01。但这块是在我下一次发送时成了 16:12 而后就再也不会去调用函数获取时间了,永远定死。这块可能的因素太多了。搞定问题暂时Run着,有分析过的老哥交流一下,peace

相关推荐
IVEN_5 小时前
只会Python皮毛?深入理解这几点,轻松进阶全栈开发
python·全栈
Ray Liang6 小时前
用六边形架构与整洁架构对比是伪命题?
java·python·c#·架构设计
AI攻城狮6 小时前
如何给 AI Agent 做"断舍离":OpenClaw Session 自动清理实践
python
千寻girling6 小时前
一份不可多得的 《 Python 》语言教程
人工智能·后端·python
AI攻城狮10 小时前
用 Playwright 实现博客一键发布到稀土掘金
python·自动化运维
曲幽10 小时前
FastAPI分布式系统实战:拆解分布式系统中常见问题及解决方案
redis·python·fastapi·web·httpx·lock·asyncio
孟健1 天前
Karpathy 用 200 行纯 Python 从零实现 GPT:代码逐行解析
python
码路飞1 天前
写了个 AI 聊天页面,被 5 种流式格式折腾了一整天 😭
javascript·python
曲幽1 天前
FastAPI压力测试实战:Locust模拟真实用户并发及优化建议
python·fastapi·web·locust·asyncio·test·uvicorn·workers