ChatGPT解决这个技术问题 Extra ChatGPT

无法比较天真和有意识的 datetime.now() <= challenge.datetime_end

我正在尝试使用比较运算符将当前日期和时间与模型中指定的日期和时间进行比较:

if challenge.datetime_start <= datetime.now() <= challenge.datetime_end:

脚本错误:

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

模型如下所示:

class Fundraising_Challenge(models.Model):
    name = models.CharField(max_length=100)
    datetime_start = models.DateTimeField()
    datetime_end = models.DateTimeField()

我也有使用语言环境日期和时间的 django。

我找不到的是 django 用于 DateTimeField() 的格式。是天真还是有意识?以及如何让 datetime.now() 识别语言环境日期时间?

有一个非常好的库可以玩日期:钟摆(我不附属)

P
Phillip

默认情况下,datetime 对象在 Python 中是 naive,因此您需要将它们都设为天真或感知 datetime 对象。这可以使用以下方法完成:

import datetime
import pytz

utc=pytz.UTC

challenge.datetime_start = utc.localize(challenge.datetime_start) 
challenge.datetime_end = utc.localize(challenge.datetime_end) 
# now both the datetime objects are aware, and you can compare them

注意:如果 tzinfo 已设置,这将引发 ValueError。如果您不确定,请使用

start_time = challenge.datetime_start.replace(tzinfo=utc)
end_time = challenge.datetime_end.replace(tzinfo=utc)

顺便说一句,您可以使用时区信息在 datetime.datetime 对象中格式化 UNIX 时间戳,如下所示

d = datetime.datetime.utcfromtimestamp(int(unix_timestamp))
d_with_tz = datetime.datetime(
    year=d.year,
    month=d.month,
    day=d.day,
    hour=d.hour,
    minute=d.minute,
    second=d.second,
    tzinfo=pytz.UTC)

它说: ValueError: Not naive datetime (tzinfo is already set) 当它尝试计算时: datetimeStart = utc.localize(challenge.datetime_start)
是的,它引发了ValueError。
替换 tzinfo 不会进行任何转换,从而导致比较不正确。
为此+1。并且,使用 utc = pytz.utc 来防止 pylint 错误 No value for argument 'dt' in unbound method call (no-value-for-parameter)pytz link
您可以从 datetimetimezone 模块获取 utc 时区。 timezone.utc
A
Alfredo Aguirre

datetime.datetime.now 不知道时区。

Django 附带了一个帮助器,它需要 pytz

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

您应该能够比较 nowchallenge.datetime_start


如果 USE_TZ=Truetimezone.now() 会返回一个可识别时区的日期时间对象,即使 pytz 未安装(尽管可能出于其他原因建议安装它)。
这应该是公认的答案
e
ePi272314

一行代码解决方案

if timezone_aware_var <= datetime.datetime.now(timezone_aware_var.tzinfo):
    pass #some code

解释版

# Timezone info of your timezone aware variable
timezone = your_timezone_aware_variable.tzinfo

# Current datetime for the timezone of your variable
now_in_timezone = datetime.datetime.now(timezone)

# Now you can do a fair comparison, both datetime variables have the same time zone
if your_timezone_aware_variable <= now_in_timezone:
    pass #some code

概括

您必须将时区信息添加到您的 now() 日期时间。
但是,您必须添加参考变量的相同时区;这就是我第一次阅读 tzinfo 属性的原因。


这似乎是最合乎逻辑的方法 - 谢谢!
H
Hiren Gohel

禁用时区。使用challenge.datetime_start.replace(tzinfo=None);

您还可以将 replace(tzinfo=None) 用于其他 日期时间

if challenge.datetime_start.replace(tzinfo=None) <= datetime.now().replace(tzinfo=None) <= challenge.datetime_end.replace(tzinfo=None):

o
ovo

没有第三方,只有本机日期时间模块。

from datetime import datetime, timedelta, timezone

time1 = datetime.strptime('2021-07-15T00:22:02+0000', '%Y-%m-%dT%H:%M:%S%z')
time2 = datetime(2021, 7, 15, tzinfo=timezone(offset=timedelta()))
if time1 < time2:
    print(True)

你将如何使用 datetime.now() 来完成这项工作?
+许多不需要第三方库!
@jtlz2,你会使用 datetime.now().replace(tzinfo=timezone(offset=timedelta()))
C
Chandan Sharma

它对我起作用。在这里,我正在创建表创建日期时间并在日期时间上添加 10 分钟。稍后根据当前时间完成到期操作。

from datetime import datetime, time, timedelta
import pytz

在数据库日期时间上增加了 10 分钟

table_datetime = '2019-06-13 07:49:02.832969'(示例)

# Added 10 minutes on database datetime
# table_datetime = '2019-06-13 07:49:02.832969' (example)

table_expire_datetime = table_datetime + timedelta(minutes=10 )

# Current datetime
current_datetime = datetime.now()


# replace the timezone in both time
expired_on = table_expire_datetime.replace(tzinfo=utc)
checked_on = current_datetime.replace(tzinfo=utc)


if expired_on < checked_on:
    print("Time Crossed)
else:
    print("Time not crossed ")

它对我有用。


m
myusuf3

所以我解决这个问题的方法是确保两个日期时间在正确的时区。

我可以看到您正在使用 datetime.now() 它将返回系统当前时间,没有设置 tzinfo。

tzinfo 是附加到日期时间的信息,让它知道它所在的时区。如果您使用的是简单的日期时间,则需要在整个系统中保持一致。我强烈建议只使用 datetime.utcnow()

看到您正在创建与 tzinfo 相关联的日期时间的某个地方,您需要做的是确保将它们本地化(与 tzinfo 相关联)到正确的时区。

看看 Delorean,它使处理这类事情变得更加容易。


您还可以使用 utcnow 看到此问题。
借调。 utcnow 不会添加时区信息
V
Vitthal Sarode

您正在尝试为已经有时区的 date_time 设置时区。使用 replaceastimezone 函数。

local_tz = pytz.timezone('Asia/Kolkata')

current_time = datetime.now().replace(tzinfo=pytz.utc).astimezone(local_tz)

H
Harispy

只是:

dt = datetimeObject.strftime(format) # format = your datetime format ex) '%Y %d %m'
dt = datetime.datetime.strptime(dt,format)

所以这样做:

start_time = challenge.datetime_start.strftime('%Y %d %m %H %M %S')
start_time = datetime.datetime.strptime(start_time,'%Y %d %m %H %M %S')

end_time = challenge.datetime_end.strftime('%Y %d %m %H %M %S')
end_time = datetime.datetime.strptime(end_time,'%Y %d %m %H %M %S')

然后使用 start_timeend_time