使用Django Rest Framework我很可爱并且玩得很开心。 但有一件事让我感到不安。我有一个模型,在这个模型中有很多洋键。即:
class Order(models.Model):
bought_by = models.ForeignKey(User, related_name='bought_orders')
bought_on = models.DateTimeField(default=datetime.datetime.now, blank=True)
category = models.ForeignKey(OrderCategory, related_name='orders')
article = models.ForeignKey(Article, related_name='orders')
supplier = models.ForeignKey(Supplier, related_name='orders')
purpose = models.CharField(null=True, blank=True, max_length=255)
payment_method = models.ForeignKey(PaymentMethod, related_name='orders')
order_number = models.CharField(max_length=255)
delivery_received_on = models.DateTimeField(null=True, blank=True)
delivery_received_by = models.ForeignKey(User, null=True, blank=True, related_name='received_orders')
tags = TaggableManager(blank=True)
所以我使用了良好的ModelSerializer,并使用嵌套的序列化器,即:
class ReadOrderSerializer(ModelSerializer):
created = DateTimeField()
updated = DateTimeField()
bought_by = UserSerializer()
category = OrderCategorySerializer()
article = ArticleSerializer()
supplier = SupplierSerializer()
payment_method = PaymentMethodSerializer()
tags = TagListSerializerField()
invoice_documents = InvoiceDocumentSerializer(many=True)
问题是请求大约需要1.7秒才能获得50个订单。 好的,这是意料之外的......我打开了我的Django调试工具并查看了sql语句。 Django Rest Framework或Django为每个订单打开一个新连接并执行select语句。
有没有办法绕过这个问题? 关闭延迟加载或我可以在序列化程序中更改的内容?
非常感谢!
答案 0 :(得分:8)
您的问题来自查询集,而不是序列化程序。默认情况下,Django查询集不遵循关系,因此不会填充相关字段。
为了解决这个问题,您必须更改查询集,以便OneToOne
和ForeignKey
字段使用select_related
,ManyToMany
字段使用prefetch_related
。
因此,假设您使用的是DRF ViewSets
,您可以将get_queryset
中的OrderViewSet
方法更改为类似的内容:
def get_queryset(self):
return Order.objects.select_related(
'bought_by', 'category', 'article', 'supplier',
'payment_method', 'delivery_received_by'
).prefetch_related(
'tags', 'invoice_documents'
)