Django Order By on Reverse外键值删除没有关系的项目

2024-09-29 01:38:58 发布

您现在位置:Python中文网/ 问答频道 /正文

上下文

我试图根据反向外键关系的值对对象列表进行排序。 我有部分解决方案,但使用的过滤器会创建一个内部连接,该连接会删除没有关系的对象

我想要用所有对象创建排序queryset的结果,就像相关值为null一样威胁“null”反向关系

型号

class Student(models.Model):
    name = models.CharField(max_length=128)


class Subject(models.Model):
    title = models.CharField(max_length=128)

class Grade(models.Model):
    student = models.ForeignKey("Student", related_name="grades", on_delete=models.CASCADE)
    subject = models.ForeignKey("Subject", related_name="grades", on_delete=models.CASCADE)
    value = models.IntegerField()

固定装置

+------+------------------------+
| name | subject  | grade_value |
+------+----------+-------------+
| beth | math     | 100         |
| beth | history  | 100         |
| mark | math     | 90          |
| mark | history  | 90          |
| mike | history  | 80          |
+------+----------+-------------+

预期结果

当根据学生的历史“成绩对学生进行排序时,我想得到[ beth (100), mark (90), mike (80) ]

现在让我们假设mark生病在家,错过了math考试

当根据学生的数学“成绩对学生进行排序时,我想得到[ beth (100), mark (90), mike (null) ]

相反,对于按数学排序的分数,返回[ beth (100), mark (90) ]

尝试的解决方案

我知道我可以获取所有结果并重新连接结果,但如果可能的话,我正在寻找一个有效的ORM查询集解决方案

>>> Student.objects.filter(grades__subject__title="history").order_by("grades__value")
<Student: Beth>, <Student: Mark>, <Student: Mike>

>>> Student.objects.filter(grades__subject__title="math").order_by("grades__value")
<Student: Beth>, <Student: Mark>

更新

我尝试按照建议使用联合,但我不确定它为什么会导致相关查找失败:

>>> Student.objects.filter(grades__subject__title="math")\
     .union(Student.objects.all())\
     .order_by("grades__value")

...
django.core.exceptions.FieldError: Cannot resolve keyword 'value' into field. Choices are: grades, id, name

溶液

下面使用@schillingt提供的解决方案进行验证

grades = Grade.objects.filter(
    student_id=OuterRef('id'), subject__title="math"
)
students = Student.objects.annotate(
    subject_grade=Subquery(grades.values('value')[:1])
)
>>> [f"{s.name}: {s.subject_grade}" for s in students.order_by("subject_grade")]
['mike: None', 'mark: 90', 'beth: 100']
>>>
>>> [f"{s.name}: {s.subject_grade}" for s in students.order_by("-subject_grade")]
['beth: 100', 'mark: 90', 'mike: None']
>>>


Tags: nameobjects排序titlevaluemodelsordermath
1条回答
网友
1楼 · 发布于 2024-09-29 01:38:58

我会在注释中使用Subquery expression

from django.db.models import Subquery, OuterRef
subject = "math"
grades = Grade.objects.filter(
    student_id=OuterRef('id'),
    subject__title=subject,
).order_by('value')
students = Student.objects.annotate(
    annotated_grade=Subquery(grades.values('value')[:1]),
)
print(students.first().annotated_grade) # The first student's highest math grade.

相关问题 更多 >