Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/python/352.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
Python Django模型:如果外键关系不存在,如何返回默认值?_Python_Django Models_Aggregation - Fatal编程技术网

Python Django模型:如果外键关系不存在,如何返回默认值?

Python Django模型:如果外键关系不存在,如何返回默认值?,python,django-models,aggregation,Python,Django Models,Aggregation,我正在与Django(德语-瑞典语)一起开发词汇培训计划 该应用程序的词汇数据由大量的“词汇卡”组成,每个“词汇卡”包含一个或多个对应于一个或多个瑞典词汇的德语单词或术语 培训仅适用于注册用户,因为应用程序通过为每个词汇卡保存分数来跟踪用户的表现 词汇卡有一个等级(基本、高级、专家)和分配给它们的任意数量的标签 当注册用户开始培训时,应用程序需要计算每个级别和标签的用户平均分数,以便用户进行选择 我通过引入名为CardByUser的模型解决了这个问题,该模型具有score和字段以及Foreign

我正在与Django(德语-瑞典语)一起开发词汇培训计划

该应用程序的词汇数据由大量的“词汇卡”组成,每个“词汇卡”包含一个或多个对应于一个或多个瑞典词汇的德语单词或术语

培训仅适用于注册用户,因为应用程序通过为每个词汇卡保存
分数来跟踪用户的表现

词汇卡有一个等级(基本、高级、专家)和分配给它们的任意数量的标签

当注册用户开始培训时,应用程序需要计算每个级别和标签的用户平均分数,以便用户进行选择

我通过引入名为
CardByUser
的模型解决了这个问题,该模型具有
score
和字段以及
ForeignKey
与模型
User
Card
的关系。现在我可以使用Django的聚合函数计算平均分数

最大的缺点是:只有当数据库中当前存在的每个卡实例都有一个
CardByUser
实例时,这种方法才有效,即使用户只训练了100张卡。我当前的解决方案是在
Card
创建和用户注册时创建所有
CardByUser
实例。当然,这在数据库内存和计算时间(注册用户需要相当长的时间)方面都是相当不有效的

而且看起来很不雅观,哪一种最让我讨厌

有更好的方法吗

在计算
卡的平均分数时,也许可以告诉Django以下内容:

  • 如果给定
    卡的
    CardByUser
    和用户存在,则使用其分数
  • 如果
    CardByUser
    不存在,请使用默认值-->分数0
这能做到吗?如果是,怎么做

编辑:澄清 谢谢S.Lott的第一个答案,但我认为我的问题有点复杂。我的缺点是,我试图用我的模型中的一些实际代码来澄清

class Card(models.Model):
    entry_sv = models.CharField(max_length=200)
    entry_de = models.CharField(max_length=200)
    ... more fields ...

class CardByUser(models.Model):
    user = models.ForeignKey(User)
    card = models.ForeignKey(Card, related_name="user_cards")
    score = models.IntegerField(default=0)
    ... more fields ...
这意味着许多
CardByUser
对象与单个
Card
相关

现在,在我的视图代码中,我需要创建满足以下条件的
CardByUser
对象的查询集:

  • 相关的
    卡片
    对象的
    标记
    字段包含某个字符串(我现在知道这也不是最优的,但不是我问题的焦点…)
  • 该用户是当前用户
然后我可以把分数加起来。我当前的代码如下(缩短):

如果当前用户的
CardByUser
,并且
Card
不存在,则它将不包括在聚合中。这就是为什么我创建了所有得分为0的
CardByUser
s

那我怎么才能摆脱这些呢?任何想法都将不胜感激

这就是方法(可能还有属性)的用途


这可能与您的问题不完全相关,但看起来,
CardByUser
确实应该是一个多对多的关系,并带有一个额外的字段。(见附件)

也许你可以这样改变你的模型

class Card(models.Model):
    entry_sv = models.CharField(max_length=200)
    entry_de = models.CharField(max_length=200)
    ... more fields ...
    users = models.ManyToManyField(User, through='CardByUser')

class CardByUser(models.Model):
    user = models.ForeignKey(User)
    card = models.ForeignKey(Card)
    score = models.IntegerField(default=0)
这样您就不必显式地创建CardByUser对象,因为这一切都由Django负责。您还应该能够简化聚合查询:

user_cards = Card.objects.filter(users=current_user)
                         .filter(tags__contains=tag.name)
...

如果只是为了getter部分,我会使用property()作为装饰器。谢谢!看起来很有希望。但有一个问题:如果卡上有新的ManyToManyField,CardByUser是否仍然需要用户的外来密钥(或者您是否忘记在代码中删除它)?如果确实需要它,我想知道为什么。无论何时使用
参数“手动”定义多对多关系,都需要它。请参阅:“当您设置中间模型时,您显式地为多个关系中涉及的模型指定外键。此显式声明定义了这两个模型的关联方式。”好的,对。但我不确定是否不必像您在回答中所写的那样显式地创建CardByUser对象。恰恰相反:文档说,这种多人关系正是通过创建中间模型的实例创建的。这正是我想要避免的。
class Card(models.Model):
    entry_sv = models.CharField(max_length=200)
    entry_de = models.CharField(max_length=200)
    ... more fields ...
    users = models.ManyToManyField(User, through='CardByUser')

class CardByUser(models.Model):
    user = models.ForeignKey(User)
    card = models.ForeignKey(Card)
    score = models.IntegerField(default=0)
user_cards = Card.objects.filter(users=current_user)
                         .filter(tags__contains=tag.name)
...