我有一个模型,比如MyModel,它有另一个模型的外键,比如Tag。
class MyModel(models.Model):
id=models.AutoField(primary_key=True)
name=models.CharField(max_length=200)
tag = models.ManyToMany(Tag)
我有大约50000个MyModel实例,每个MyModel可以有100个标签。
如果我使用上面的模型,我将在一个表mymodel_tag中获得5000000个条目,但我可以使用ORM的所有功能。
然而,如果我编写自定义方法并将上面的字段视为整数数组,并编写自定义代码来检索/保存与MyModel相关的标签的ID,我将只有50K个条目,但我将不得不编写用于检索等的自定义代码。
a) 我想知道这两种方法的利弊!
b) 如果我必须使用自定义数组方法,我如何才能有效地做到这一点。
嗯。。
tag = models.ManyToManyField(Tag)
对于外键,MyModel
只能与一个且只能与一个子Tag
相关联。老实说,我甚至不知道你是如何给每个人100个Tags
而不必重复每个MyModel
100次的。如果你这样做,难怪你不喜欢结果。
ManyToManyField
创建一个联接表,该联接表将仅由对MyModel
的id(integer)引用和对Tag
的id(整数)引用组成。这是你在这种关系中得到的最紧凑的关系,无论如何,这是最好的做法。
虽然我完全同意chrisdpratt的说法,但不幸的是,我被迫采取其他方式。以下是我在http://djangosnippets.org/snippets/1200/:
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)