Django自定义模型管理器数据库错误

时间:2012-05-15 06:38:10

标签: django django-models

我正在尝试构建自定义模型管理器,但遇到了错误。代码如下所示:

class LookupManager(models.Manager):
    def get_options(self, *args, **kwargs):
        return [(t.key, t.value) \
                for t in Lookup.objects.filter(group=args[0].upper())]


class Lookup(models.Model):
    group = models.CharField(max_length=1)
    key = models.CharField(max_length=1)
    value = models.CharField(max_length=128)
    objects = LookupManager()

(我使用get_options以及过滤结果的其他方式使用super()玩了很多次)

当我运行syncdb时,我收到以下错误(ops_lookup是相应的表格):

django.db.utils.DatabaseError: no such table: ops_lookup

我注意到如果我更改管理器以返回[]而不是过滤器,则syncdb可以正常工作。此外,如果我运行syncdb并且所有表都存在,那么将代码更改为上面的代码,它也可以。

首次运行syncdb时,如何让Django不要指望此表存在?

更新 在回顾了追溯之后,我意识到发生了什么。查找表旨在包含填充其他表中某些列的选项的值。我认为发生的事情是在创建其他表时调用管理器,它似乎发生在查找表创建之前。

有没有办法强制django首先创建查找表(没有重命名它?)

1 个答案:

答案 0 :(得分:0)

正在发生的是您在模块加载期间尝试访问数据库。例如:

class MyModel(models.Model):
    name = models.CharField(max_length=255)


class OtherModel(models.Model):
    some_field = models.CharField(
        max_length=255,
        # Next line fails on syncdb because the database table hasn't been created yet
        # but the model is being queried during module load time (during class definition)
        choices=[(o.pk, o.name) for o in MyModel.objects.all()]
    )

这相当于你正在做的事情,因为正如你所说,你正在使用管理员方法(传递)为其他模型生成选择。

使用生成器表达式替换列表解析将返回可迭代,但在第一次迭代之前不会评估已过滤的查询集。所以,这将修复上面的例子:

choices=((o.pk, o.name) for o in MyModel.objects.all())

使用您的示例,它将是:

class LookupManager(models.Manager):
    def get_options(self, *args, **kwargs):
        return ((t.key, t.value) for t in Lookup.objects.filter(group=args[0].upper()))

(请注意使用()代替[])(外部代码) - 这是创建生成器表达式的语法。< / p>