Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/django/19.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Django:多对多的自定义表示_Django_Django Models_Foreign Keys - Fatal编程技术网

Django:多对多的自定义表示

Django:多对多的自定义表示,django,django-models,foreign-keys,Django,Django Models,Foreign Keys,我有一个模型,比如说MyModel,它有另一个模型的外键,比如说标记 class MyModel(models.Model): id=models.AutoField(primary_key=True) name=models.CharField(max_length=200) tag = models.ManyToMany(Tag) 我有大约50000个MyModel实例,每个MyModel可以有100个标记 如果我使用上述模型,我将在一个表中获得5000000个条目

我有一个模型,比如说MyModel,它有另一个模型的外键,比如说标记

class MyModel(models.Model):
    id=models.AutoField(primary_key=True)
    name=models.CharField(max_length=200)
    tag = models.ManyToMany(Tag)
我有大约50000个MyModel实例,每个MyModel可以有100个标记

如果我使用上述模型,我将在一个表中获得5000000个条目,mymodel_标记,但我可以使用ORM的所有功能

但是,如果我编写自定义方法并将上述字段视为整数数组,并编写自定义代码来检索/保存与MyModel关联的标记ID,我将只有50K个条目,但我必须编写用于检索等的自定义代码

a) 我想知道这两种方法的利弊

b) 如果我必须使用自定义数组方法,我如何才能有效地使用它。

tag = models.ManyToManyField(Tag)
?

使用外键,
MyModel
只能与一个且只有一个
标记关联。老实说,我甚至不知道你怎么能给每个人100个
标签
,而不必复制每个
MyModel
100次。如果你这么做,难怪你不喜欢结果


ManyToManyField
创建一个联接表,该联接表将只包含对
MyModel
的id(整数)引用和对
标记的id(整数)引用。这是这种关系中最紧密的一次,无论如何,这是最好的做法。

虽然我完全同意克里斯普拉特的说法,但不幸的是,我不得不以其他方式去做。以下是我在以下网站上找到的一种方法:


你完全正确,我的意思是ManyToMany(这就是为什么也提到了join表),但最终以外键的形式编写。但问题是,拥有一个包含500万行的联接表更好,还是我应该将其保存为另一列中的ID数组?如果我使用另一个专栏,那么最好的方法是什么。谢谢联接表将被索引,所以我认为行的数量并不重要。一般来说,遵循最佳实践始终是最佳实践,而且许多ManyToManyField的行为与Django中的行为相同,因为软件开发人员几乎普遍认为联接表是处理多对多关系的最佳方式。
from django.db import models
from django import forms

class MultiSelectFormField(forms.MultipleChoiceField):
    widget = forms.CheckboxSelectMultiple

    def __init__(self, *args, **kwargs):
        self.max_choices = kwargs.pop('max_choices', 0)
        super(MultiSelectFormField, self).__init__(*args, **kwargs)

    def clean(self, value):
        if not value and self.required:
            raise forms.ValidationError(self.error_messages['required'])
        if value and self.max_choices and len(value) > self.max_choices:
            raise forms.ValidationError('You must select a maximum of %s choice%s.'
                    % (apnumber(self.max_choices), pluralize(self.max_choices)))
        return value

class MultiSelectField(models.Field):
    __metaclass__ = models.SubfieldBase

    def get_internal_type(self):
        return "CharField"

    def get_choices_default(self):
        return self.get_choices(include_blank=False)

    def _get_FIELD_display(self, field):
        value = getattr(self, field.attname)
        choicedict = dict(field.choices)

    def formfield(self, **kwargs):
        # don't call super, as that overrides default widget if it has choices
        defaults = {'required': not self.blank, 'label': capfirst(self.verbose_name), 
                    'help_text': self.help_text, 'choices':self.choices}
        if self.has_default():
            defaults['initial'] = self.get_default()
        defaults.update(kwargs)
        return MultiSelectFormField(**defaults)

    def get_db_prep_value(self, value):
        if isinstance(value, basestring):
            return value
        elif isinstance(value, list):
            return ",".join(value)

    def to_python(self, value):
        if isinstance(value, list):
            return value
        return value.split(",")

    def contribute_to_class(self, cls, name):
        super(MultiSelectField, self).contribute_to_class(cls, name)
        if self.choices:
            func = lambda self, fieldname = name, choicedict = dict(self.choices):",".join([choicedict.get(value,value) for value in getattr(self,fieldname)])
            setattr(cls, 'get_%s_display' % self.name, func)