将django RawQuerySet转换为Queryset

时间:2018-01-16 18:58:33

标签: python django postgresql django-models django-managers

我有2个Django模型,ModelAArrayField,用于存储大量主键值列表(可能是50k +列表)

class ModelA(models.Model):
    pk_values = ArrayField(models.IntegerField())

class CustomManager(manager.Manager):

    def get_for_index(self, index_id):
        qs = self.get_queryset()
        obj = ModelA.objects.get(pk=index_id)
        return qs.filter(id__in=obj.pk_values)

class ModelB(models.Model):
    # [...] some fields

    objects = CustomManager()

这有效:

qs = ModelB.objects.get_for_index(index_id=1)

但是,如果“pk_values”是一个很大的列表,这将会非常慢。

所以我尝试进行原始SQL查询:

class CustomManager(manager.Manager):
    def get_for_index(self, index_id):
        qs = self.get_queryset()
        sql = "SELECT * FROM myapp_model_b JOIN myapp_model_a ON myapp_model_b.id = ANY(myapp_model_a.pk_values) WHERE myapp_model_a.id = '%s'" % index_id
        return qs.raw(sql)

但这会返回django.db.models.query.RawQuerySet个实例。

但有了这个,我之后就不能做queryset.values()之类的事了。

如何将其转换为普通的Django查询集?

有更好的方法吗?

文档:

2 个答案:

答案 0 :(得分:11)

您可以使用RawSQL表达式:

ModelB.objects.filter(id__in=RawSQL(
    'SELECT unnest(a.pk_values) FROM app_modela a WHERE a.id = %s',
    [index_id]
))

或者,您可以使用extra()

重现问题中的确切查询
ModelB.objects.extra(
    tables=['foo_modela'],
    where=[
        '"app_modelb"."id" = ANY("app_modela"."pk_values")',
        '"app_modela"."id" = %s',
    ],
    params=[index_id],
)

答案 1 :(得分:1)

更新:我使用.extra()

工作了
class CustomManager(manager.Manager):
    def get_for_index(self, index_id):
        qs = self.get_queryset()
        sql = "myapp_model_b.id IN (SELECT UNNEST(myapp_model_a.pk_values) FROM myapp_model_a WHERE myapp_model_a.id='%s')" % index_id
        return qs.extra(where=[sql])

文档:https://docs.djangoproject.com/en/2.0/ref/models/querysets/#django.db.models.query.QuerySet.extra