Django ORM:根据相关模型的时间顺序字段过滤主模型

时间:2015-10-08 13:17:57

标签: python django

让我们假设我们有以下模型:

class Patient(models.Model)
    name = models.CharField()
    # other fields following

class MedicalFile(model.Model)
    patient = models.ForeignKey(Patient, related_name='files')
    date = models.DateField()
    diagnostic = models.CharField()

我们希望为患者构建一个视图集,我们希望根据患者记录的最新可用/有效诊断来过滤患者记录。

我不知道如何在不使用原始SQL的情况下解决这个问题。是否有使用Django查询集语法构建此语句的最佳方法?

<小时/> 我是如何解决这个问题的?

我确信这不太好,但它最适合过滤大型数据集。

我们的想法是使用在数据库层实现的视图,我们将查询所有患者及其相关的最新医疗文件,而不是将Django实体映射到该视图。对于科西嘉,我们将使新模型成为无管理的。

为什么走这么长的路?因为在新模型上我们可以使用“可重用”的Django查询语法。对于corse,数据库中的视图不可重用,必须为每个db后端解决方案重新创建。

考虑到Postgres,这将是视图定义:

SELECT
    p.*,
    f.*
FROM Person p
    LEFT JOIN (
        SELECT
            *,
            max(date) OVER (PARTITION BY person_id) AS latest_date
        FROM MedicalFile
    ) AS mf ON mf.person_id = p.person_id
WHERE
    mf.latest_date IS NULL OR mf.latest_date = mf.date

然后我们可以创建像这样的关联模型

class LatestMedicalFile(models.Model):
    patient = models.OneToOneField(Patient, related_name="latest_file")
    date = models.DateField()
    diagnostic = models.CharField()

    class Meta:
        managed = False
        db_table = '<your view name here>'

最后,我们的查询可以这样写:

Patient.objects.filter(latest_file__diagnostic='flu')

在我看来,这不直观,也不干净。有什么想法吗?

1 个答案:

答案 0 :(得分:1)

如果您有兴趣查看文件:

MedicalFile.objects.annotate(
    maxdate=Max(
        'patient__files__date'
        )).filter(
            maxdate=F('date'),
            diagnosis="flu").select_related(
                'patient'
            )

如果你想要患者:

Patient.objects.annotate(
    maxdate=Max(
        'files__date'
        )).filter(
            maxdate=F('files__date'),
            files__diagnosis="flu"))

非常感谢Roba,协作提问/回答是我最喜欢的SO用户类型。