如何在 Django 模板中比较日期
- 2025-03-20 08:48:00
- admin 原创
- 47
问题描述:
我想在 Django 中将日期与当前日期进行比较,最好是在模板中,但也可以在渲染模板之前进行。如果日期已经过去,我想说“过去”,而如果它是将来的,我想给出日期。
我希望有人能做这样的事:
{% if listing.date <= now %}
In the past
{% else %}
{{ listing.date|date:"d M Y" }}
{% endif %}
现在是今天的日期,但这不起作用。我在 Django 文档中找不到有关此内容的任何信息。有人能提供一些建议吗?
解决方案 1:
比较视图中的日期,并将类似in_the_past
(布尔值)的内容传递给 extra_context。
或者最好将其作为属性添加到模型中。
from datetime import date
@property
def is_past_due(self):
return date.today() > self.date
然后在模板中:
{% if listing.is_past_due %}
In the past
{% else %}
{{ listing.date|date:"d M Y" }}
{% endif %}
基本上,模板不是用于日期比较的地方。
解决方案 2:
从 Django 1.8 开始,下面稍微有点令人反感的构造可以完成这项工作:
{% now "Y-m-d" as todays_date %}
{% if todays_date < someday|date:"Y-m-d" %}
<h1>It's not too late!</h1>
{% endif %}
有点黑客风格,但是它避免了自定义标签或上下文处理器的需要。
解决方案 3:
我将 date_now 添加到我的上下文处理器列表中。
因此在模板中有一个名为“date_now”的变量,它只是 datetime.datetime.now()
在文件 context_processors.py 中创建一个名为 date_now 的上下文处理器
import datetime
def date_now(request):
return {'date_now':datetime.datetime.now()}
在 settings.py 中,修改 CONTEXT_PROCESSORS 以包含它,在我的情况下它是
app_name.context_processors.date_now
解决方案 4:
除了@bx2有益的答案之外,如果您的字段是日期时间字段,只需调用date()函数来对日期时间字段进行建模:
from datetime import date
@property
def is_past_due(self):
if date.today() > self.date.date():
return True
return False
编辑:我认为这可以缩小到:
from datetime import date
@property
def is_past_due(self):
return date.today() > self.date.date()
解决方案 5:
我相信实现此目的的最简单方法是在 views.py 中导入日期时间并将今天的日期作为上下文数据传递。
context['today'] = datetime.date.today()
然后在模板标签中您可以执行类似的操作。
{% if listing.date < today % }
这样,如果您将对象列表作为上下文传递,则可以在 HTML 模板中输出时将过滤器应用于每一行。我有一个已过滤掉的项目列表,并在显示它们时使用 Bootstrap 进行样式化。我想让逾期日期脱颖而出,并且仅当我的日期之一小于今天的日期时才应用过滤。
解决方案 6:
您始终可以传递 datetime.datetime.now(因为 django 模型使用 Python 的标准 datetime 对象)。
使用 render_to_response,您可以执行以下操作(导入日期时间后):
return render_to_response('template.html', {'now': datetime.datetime.now()})
现在您可以访问模板内的“现在”,您可以像在示例中一样比较日期。
此外,如果您在视图中使用 RequestContext - 您将能够将“now”添加为 context_processor(如果您需要在多个文件中使用此功能)。这会将“now”添加到使用 RequestContext 呈现的任何模板中。
然而,更现实的是,你只需获取原始查询集中之前的记录列表,并避免查询无用的数据:
listing.objects.filter(date__lt=datetime.datetime.now())
解决方案 7:
您可以有 2 项任务。
您必须使用 DetailView 显示对象。您可以使用将布尔值传递给模板的函数来解决此问题。但是,请记住,在 django 中,您无法将 SQL 数据库中的日期与 进行比较
datetime.datetime.now()
,因为第二个选项不包含时区信息。请使用django.utils.timezone.now()
。您必须从 ListView 中设置对象的样式。然后我建议您使用将在模板中调用的自定义过滤器。
@register.filter
def compare_date(date):
if not date:
return False
return date < timezone.now()
解决方案 8:
如果你使用 Django,你可以使用下面的代码(https://stackoverflow.com/a/3798865/17050678)。在你的 models.py 中
@property
def is_past_due(self):
today = datetime.datetime.today()
return today > self.start_date
但这会给你带来一个错误。就像这样:
can't compare offset-naive and offset-aware datetimes
因此您应该采用如下所示的时区类型。
import pytz
@property
def is_past_due(self):
today = datetime.datetime.today().astimezone(pytz.timezone('Asia/Tokyo'))
return today > self.start_date
然后您可以在模板中进行比较。
{% if schedule.is_past_due %}
/*your code here*/
{% endif %}
解决方案 9:
我找到了这个问题,也遇到了类似的问题。我希望显示过去发生过的信息。
使用模板中的 Django 的“timesince”标签,我能够解决我的问题。我并不是说它很高效,但它对我来说很有效。
在模板中:
{% if model_name.date_time|timesince >= "1 min" %}
<p>In the past</p>
{% else %}
<p>{{ model_name.date_time }}</p>
{% endif %}
扫码咨询,免费领取项目管理大礼包!