18

使用 Django 1.5.1。蟒蛇 2.7.3。

我想用一个外键字段和一个 slug 字段做一个唯一的共同约束。所以在我的模型元中,我做到了

foreign_key = models.ForeignKey("self", null=True, default=None)
slug = models.SlugField(max_length=40, unique=False)

class Meta:
    unique_together = ("foreign_key", "slug")

我什至检查了 Postgres (9.1) 中的表描述,并将约束放入数据库表中。

-- something like
"table_name_foreign_key_id_slug_key" UNIQUE CONSTRAINT, btree (foreign_key_id, slug)

但是,我仍然可以将 None/null 和重复字符串的 foreign_key 保存到数据库表中。

例如,

我可以输入并保存

# model objects with slug="python" three times; all three foreign_key(s) 
# are None/null because that is their default value
MO(slug="python").save()
MO(slug="python").save()
MO(slug="python").save()

那么在使用 unique_together 之后,为什么我仍然可以输入三个相同值的行呢?

我现在只是猜测它可能与foreign_key字段的默认值None有关,因为在unique_together之前,当我在slug上只有unique = True时,一切正常。因此,如果是这种情况,我应该使用什么默认值来指示空值,同时保持唯一约束?

4

4 回答 4

25

在 PostgresqlNULL中不等于任何其他NULL. 因此,您创建的行是不同的(从 Postgres 的角度来看)。

更新

你有几种方法来处理它:

  • 禁止Null外键的值并使用一些默认值
  • 覆盖save模型的方法以检查不存在这样的行
  • 更改 SQL 标准 :)
于 2013-07-07T07:59:38.333 回答
1

向模型添加clean方法,以便您可以编辑现有行。

def clean(self):
    queryset = MO.objects.exclude(id=self.id).filter(slug=self.slug)
    if self.foreign_key is None:
        if queryset.exists():
            raise ValidationError("A row already exists with this slug and no key")
    else:
        if queryset.filter(foreign_key=self.foreign_key).exists():
            raise ValidationError("This row already exists")

请注意,clean(or full_clean) 不会被默认save方法调用。

注意:如果您将此代码放入save方法中,更新表单(如在管理员中)将不起作用:由于ValidationError异常,您将遇到回溯错误。

于 2013-10-29T23:19:19.667 回答
0

只需在字段上手动创建二级索引slug,但仅限于 NULL 值foreign_key_id

CREATE INDEX table_name_unique_null_foreign_key
  ON table_name (slug) WHERE foreign_key_id is NULL

请注意,Django 不支持这一点,因此如果没有自定义表单/模型验证,您将得到纯 IntegrityError / 500。

使用空列创建唯一约束的可能重复项

于 2015-10-20T20:44:53.710 回答
0

正如 hobbyte 所提到的,“在 Postgresql 中,NULL 不等于任何其他 NULL。因此,您创建的行是不一样的(从 Postgres 的角度来看)。”

解决这一挑战的另一种可能方法是在 form_valid 方法中的视图级别添加自定义验证。

在views.py中:

def form_valid(self, form): 

  --OTHER VALIDATION AND FIELD VALUE ASSIGNMENT LOGIC--

  if ModelForm.objects.filter(slug=slug,foreign_key=foreign_key:   
    form.add_error('field',
      forms.ValidationError( _("Validation error message that shows up in your form. "), 
      code='duplicate_row', )) 
    return self.form_invalid(form)

如果您使用基于类的视图,这种方法很有帮助,尤其是当您自动为要对用户隐藏的字段分配值时。

优点:

  • 您不必在数据库中创建虚拟默认值
  • 您仍然可以使用更新表格(请参阅 Toff 的回答)

缺点: - 这不能防止直接在数据库级别创建的重复行。- 如果您使用 Django 的管理后端来创建新的 MyModel 对象,则需要将相同的验证逻辑添加到您的管理表单中。

于 2016-06-09T23:18:46.020 回答