使用'through'时,'ManyToManyField`的重复结果

时间:2015-01-05 09:58:00

标签: django many-to-many django-database

我有两种模式:用户文章 每个用户都可以通过选择一个或多个标记来创建文章

我在我的模型中使用以下表示:

class User(models.Model):
    name = models.CharField(max_length=200, null=False, blank=False, default=None)
    articles = models.ManyToManyField(Article, null=True, blank=True, default=None, through='Tag')

class Article(models.Model):
    url = models.URLField(unique=True)

class Tag(models.Model):
    name = models.CharField(max_length=200, null=False, blank=False, default=None)
    article = models.ForeignKey(Article)
    user = models.ForeignKey(User)

现在,我可以单独添加用户和文章。然后我可以将用户与文章相关联。 假设我只有<user 1><article 1>。我创建了两个不同的标签(通过管理员完成):

          Tag1
<user 1>  --->   <article 1>
          Tag2
<user 1>  --->   <article 1>

我仍然只有一个用户,只有一篇文章,但有两个标签正在链接它们。 当我查询user1.article.all()时,我希望只有一篇文章作为结果。

预期行为

user = User.objects.get(pk=1)
print user.article.all()
[<Article: 1>]

观察到的行为

user = User.objects.get(pk=1)
print user.article.all()
[<Article: 1>, <Article: 1>]

任何人都知道这是一个错误,还是我以错误的方式查询Django(1.7.1和1.7.2)? 如果这是正常的,我该如何轻松删除重复项?

[编辑]在错误发生时打开一张票:https://code.djangoproject.com/ticket/24079#ticket

1 个答案:

答案 0 :(得分:2)

您需要使用.distinct()。来自docs

  

默认情况下,QuerySet不会消除重复的行。实际上,这很少是一个问题,因为Blog.objects.all()之类的简单查询不会引入重复结果行的可能性。但是,如果您的查询跨越多个表,则在评估QuerySet时可能会获得重复的结果。那是你使用distinct()的时候。