我在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(尽管这样做更烦人)。


当前回答

您是否尝试删除时区识别?

从http://pytz.sourceforge.net/

naive = dt.replace(tzinfo=None)

可能还得加上时区转换。

编辑:请注意这个答案的年龄。下面是一个涉及添加时区信息而不是在python3中删除它的答案。https://stackoverflow.com/a/25662061/93380

其他回答

标题中对这个问题最直接的回答似乎缺失了:

naive_date.replace(tzinfo=aware_date.tzinfo) - aware_date

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

select *, age(timeStampField) as timeStampAge from myTable

您是否尝试删除时区识别?

从http://pytz.sourceforge.net/

naive = dt.replace(tzinfo=None)

可能还得加上时区转换。

编辑:请注意这个答案的年龄。下面是一个涉及添加时区信息而不是在python3中删除它的答案。https://stackoverflow.com/a/25662061/93380

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

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

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

你不需要性病资料库以外的任何东西

datetime.datetime.now().astimezone()

如果你只是更换了时区,它不会调整时间。如果你的系统已经是UTC,那么.replace(tz='UTC')就可以了。

>>> x=datetime.datetime.now()
datetime.datetime(2020, 11, 16, 7, 57, 5, 364576)

>>> print(x)
2020-11-16 07:57:05.364576

>>> print(x.astimezone()) 
2020-11-16 07:57:05.364576-07:00

>>> print(x.replace(tzinfo=datetime.timezone.utc)) # wrong
2020-11-16 07:57:05.364576+00:00