让我们假设我们有以下模型:
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 查询集语法构建此语句的最佳方法?
How did I solve this ?
我确信这不太好,但它对于过滤大型数据集来说是最佳的。
我们的想法是使用在数据库层实现的视图,我们将在其中查询所有患者及其相关的最新医疗文件,而不是将 Django 实体映射到该视图。当然,我们将使新模型成为非托管的。
为什么要走这么远的路?因为在新模型上我们可以使用“可重用”的 Django 查询语法。当然,数据库中的视图不可重复使用,必须为每个数据库后端解决方案重新创建。
考虑到 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')
在我看来,这既不直观,也不干净。有任何想法吗 ?