Как сделать обратный фильтр на основе последней записи в Django
У меня есть таблица клиентов и таблица заказов клиентов, как указано ниже
class Customer(models.Model):
name = models.CharField()
class CustomerOder(models.Model):
customer = models.ForiengKey(Customer)
item = models.CharField()
datetime_of_purchase = models.DateField()
Образец записи в базе данных
Таблица: Customer
id name
1 A
2 B
3 C
Таблица: CustomerOrder
id customer item datetime_of_purchase
1 A item_3 2021-11-10 10:00:00
2 A item_2 2021-11-11 10:00:00
3 B item_1 2021-11-11 10:00:00
4 A item_1 2021-11-12 10:00:00
5 B item_2 2021-11-12 10:00:00
6 C item_1 2021-11-13 10:00:00
Предположим, мне нужно отфильтровать клиентов, чья последняя покупка - 'items_1', поэтому выборка после фильтрации должна получить только клиентов 'A' и 'C', чья последняя покупка - 'item_1'
Sample:
<QuerySet [<Customer: A>, <Customer: C>]>
Вы можете получить последний вычисленный элемент с помощью выражения Subquery
[Django-doc]:
from django.db.models import OuterRef, Subquery
Customer.objects.alias(
latest_purchase=Subquery(
CustomerOrder.objects.filter(
customer_id=OuterRef('pk')
).order_by('-datetime_of_purchase').values('item')[:1]
)
).filter(latest_purchase='item_1')
Перед django-3.2, используется .annotate(…)
[Django-doc] вместо .alias(…)
[Django-doc]:
from django.db.models import OuterRef, Subquery
Customer.objects.annotate(
latest_purchase=Subquery(
CustomerOrder.objects.filter(
customer_id=OuterRef('pk')
).order_by('-datetime_of_purchase').values('item')[:1]
)
).filter(latest_purchase='item_1')