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


当前回答

我知道有些人专门使用Django作为接口来抽象这种类型的数据库交互。Django提供了一些实用工具:

from django.utils import timezone
now_aware = timezone.now()

你需要建立一个基本的Django设置基础设施,即使你只是使用这种类型的接口(在设置中,你需要包括USE_TZ=True来获得一个感知datetime)。

就其本身而言,这可能远远不足以激励您使用Django作为接口,但还有许多其他好处。另一方面,如果您在这里遇到问题是因为您正在破坏Django应用程序(就像我一样),那么这可能会有所帮助……

其他回答

我知道有些人专门使用Django作为接口来抽象这种类型的数据库交互。Django提供了一些实用工具:

from django.utils import timezone
now_aware = timezone.now()

你需要建立一个基本的Django设置基础设施,即使你只是使用这种类型的接口(在设置中,你需要包括USE_TZ=True来获得一个感知datetime)。

就其本身而言,这可能远远不足以激励您使用Django作为接口,但还有许多其他好处。另一方面,如果您在这里遇到问题是因为您正在破坏Django应用程序(就像我一样),那么这可能会有所帮助……

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

从http://pytz.sourceforge.net/

naive = dt.replace(tzinfo=None)

可能还得加上时区转换。

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

这是一个非常简单明了的解决方案——两行代码

# First we obtain de timezone info o some datatime variable    

tz_info = your_timezone_aware_variable.tzinfo

# Now we can subtract two variables using the same time zone info
# For instance
# Lets obtain the Now() datetime but for the tz_info we got before

diff = datetime.datetime.now(tz_info)-your_timezone_aware_variable

结论:必须使用相同的时间信息来管理datetime变量

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

select *, age(timeStampField) as timeStampAge from myTable

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