正确生成Django模型的唯一标识符

时间:2013-05-11 17:52:54

标签: django django-models

我在model.py中执行了以下操作,为其生成唯一标识符:

 12 def rand_key(size):
 13     return ''.join([random.choice(string.letters + string.digits) for i in range(size)])
 14 
 15 
 16 class Share(models.Model):
 17 
 20     resource_content_type = models.ForeignKey(ContentType)
 21     resource_object_id = models.PositiveIntegerField()
 22     resource_content_object = generic.GenericForeignKey('resource_content_type', 'resource_object_id')
 23 
 24     identifier = models.CharField(max_length=16,unique=True,default=rand_key(16))

问题在于,当尝试连续保存两条记录时,会弹出错误消息duplicate key value violates unique constraint "share_share_identifier_key"。我确信这是一个常见的功能。

我做错了什么?

由于

编辑:让我添加创建的第一条记录没有任何问题,标识符字段具有一个比较“随机”的值。是在发生冲突的第二个记录的cration。似乎rand_key函数以某种方式被缓存。

2 个答案:

答案 0 :(得分:5)

问题是您将随机函数的输出分配给默认参数。然后,所有条目的值都相同。试试这段代码

def rand_key(size):
    return ''.join([random.choice(string.letters + string.digits) for i in range(size)])


class Share(models.Model):
    resource_content_type = models.ForeignKey(ContentType)
    resource_object_id = models.PositiveIntegerField()
    resource_content_object = generic.GenericForeignKey('resource_content_type', 'resource_object_id')
    identifier = models.CharField(max_length=16, unique=True)

    def save(self, force_insert=False, force_update=False, using=None,
         update_fields=None):
        if self.identifier is None:
            self.identifier = rand_key(16)
        models.Model.save(self, force_insert, force_update, using, update_fields)

每次保存条目时,这将生成随机字符。

答案 1 :(得分:2)

对Muhammed K K的答案进行了几次更正(会有评论但不允许)。这将覆盖模型保存方法,并在第一次保存模型时添加唯一ID。

def save(self, force_insert=False, force_update=False, using=None,
         update_fields=None):
        if self.identifier is None or len(self.identifier) == 0:
            self.identifier = rand_key(16)
        models.Model.save(self, force_insert, force_update, using, update_fields)

我需要传递self作为保存的第一个参数,并检查标识符是否为空字符串OR null。我不知道导致这些差异的原因是什么,或者他的答案是不是要使用vertatim ...我正在使用Django 1.6和python 2.6。