2

我使用 django 序列化程序序列化了几个对象,但问题是每个序列化都从 db 查询 foerign 键的 id,而不是仅仅从对象中获取它,例如

class QBAccount(CompanyModel):
    company = models.ForeignKey(Company)

>>> from deretoapp.models import QBAccount
>>> import logging
>>> l = logging.getLogger('django.db.backends')
>>> l.setLevel(logging.DEBUG)
>>> l.addHandler(logging.StreamHandler())
>>> a = QBAccount.allobjects.all()[0]
>>> from django.core import serializers
>>> serializers.serialize('python', [a])
(0.000) SELECT `deretoapp_company`.`id`, ... FROM `deretoapp_company` WHERE `deretoapp_company`.`id` = 45995613-adeb-488f-9556-d69e856abe5f ; args=(u'45995613-adeb-488f-9556-d69e856abe5f',)
[{'pk': u'3de881eb-8409-4089-8de8-6e24f7281f37', 'model': u'deretoapp.qbaccount', 'fields': {... 'company': u'45995613-adeb-488f-9556-d69e856abe5f' ....}}]

有没有办法在不修改 django 代码的情况下改变这种行为?我知道a.company.id会查询公司表(这在理想世界中不应该发生)但是序列化程序中有一个选项,以便它执行类似a.company_id不会查询数据库的操作

>>> django.VERSION
(1, 3, 1, 'final', 0)
4

3 回答 3

1

我最终修改了 django python 序列化程序,以便它直接获取引用对象的 id 而不是从 db 获取它

from django.core.serializers.python import Serializer as PythonSerializer
from django.core.serializers.python import Deserializer

class Serializer(PythonSerializer):
    internal_use_only = False

    def handle_fk_field(self, obj, field): 
        if not self.use_natural_keys:
            # directly get the id
            self._current[field.name] = getattr(obj, field.attname)
            return

        return super(Serializer, self).handle_fk_field(obj, field)

我不确定它是否会处理所有用ForeighKey例,但它适用于简单的情况,例如company = models.ForeignKey(Company)

还需要在settings.py中注册序列化器

SERIALIZATION_MODULES = { 'python' : 'myapp.serializers.python' }

我还为此提交了一个错误 ,现在已在 django 主干中修复。 见变更集

于 2012-01-27T17:40:49.160 回答
0

您应该使用Natural Keys进行序列化。

本质上,这需要您定义一个在外键引用模型上调用的方法natural_key,该方法返回您希望拥有的字段,代替 pk

从文档中:

class Person(models.Model):
    objects = PersonManager()

    first_name = models.CharField(max_length=100)
    last_name = models.CharField(max_length=100)

    birthdate = models.DateField()

    def natural_key(self):
        return (self.first_name, self.last_name)

    class Meta:
        unique_together = (('first_name', 'last_name'),)

然后在序列化过程中将参数设置use_natural_keys为。True

同样,从文档中:

>>> serializers.serialize('json', [book1, book2], indent=2, use_natural_keys=True)
于 2012-01-27T16:09:44.683 回答
0

我为避免额外查询所做的工作似乎也不是很有效,但它确实有效:添加select_related(...)到原始查询集中以在初始查询中加载相关对象。

例如:

a = QBAccount.allobjects.select_related('company').all()[0]
于 2013-06-09T12:28:17.067 回答