我在PostgreSQL中有一个时区感知timestamptz字段。当我从表中提取数据时,我想减去现在的时间,这样我就可以得到它的年龄。

我遇到的问题是datetime.datetime.now()和datetime.datetime.utcnow()似乎都返回不知道时区的时间戳,这导致我得到这个错误:

TypeError: can't subtract offset-naive and offset-aware datetimes 

是否有办法避免这种情况(最好不使用第三方模块)。

编辑:谢谢你的建议,但试图调整时区似乎给我错误。所以我将在PG中使用不考虑时区的时间戳,并且总是插入using:

NOW() AT TIME ZONE 'UTC'

这样,默认情况下我的所有时间戳都是UTC(尽管这样做更烦人)。


当前回答

我也面临着同样的问题。经过一番寻找,我终于找到了解决办法。

问题是,当我们从模型或表单获取datetime对象时,它是偏移量感知的,如果我们通过系统获取时间,它是偏移量无知的。

因此,我所做的就是使用timezone.now()获取当前时间,并从django导入timezone by。utils导入时区,并将USE_TZ = True放在项目设置文件中。

其他回答

psycopg2模块有自己的时区定义,所以我最终编写了自己的utcnow包装器:

def pg_utcnow():
    import psycopg2
    return datetime.utcnow().replace(
        tzinfo=psycopg2.tz.FixedOffsetTimezone(offset=0, name=None))

当你需要将当前时间与PostgreSQL时间戳进行比较时,只需使用pg_utcnow

我想到了一个超简单的解决办法:

import datetime

def calcEpochSec(dt):
    epochZero = datetime.datetime(1970,1,1,tzinfo = dt.tzinfo)
    return (dt - epochZero).total_seconds()

它既适用于时区感知型datetime值,也适用于时区无知型datetime值。并且不需要额外的库或数据库解决方案。

我也面临着同样的问题。经过一番寻找,我终于找到了解决办法。

问题是,当我们从模型或表单获取datetime对象时,它是偏移量感知的,如果我们通过系统获取时间,它是偏移量无知的。

因此,我所做的就是使用timezone.now()获取当前时间,并从django导入timezone by。utils导入时区,并将USE_TZ = True放在项目设置文件中。

我发现timezone.make_aware(datetime.datetime.now())在django中很有帮助(我在1.9.1)。不幸的是,你不能简单地让一个datetime对象感知偏移量,然后timez()它。你必须制定一个日期时间,并在此基础上进行比较。

有什么紧迫的原因,为什么你不能处理年龄计算PostgreSQL本身?类似的

select *, age(timeStampField) as timeStampAge from myTable