创建 Django 模型或更新(如果存在)

Posted

技术标签:

【中文标题】创建 Django 模型或更新(如果存在)【英文标题】:Create Django model or update if exists 【发布时间】:2012-12-16 10:08:50 【问题描述】:

我想创建一个模型对象,比如 Person,如果人的 id 不存在,或者我会得到那个人对象。

新建人物的代码如下:

class Person(models.Model):
    identifier = models.CharField(max_length = 10)
    name = models.CharField(max_length = 20)
    objects = PersonManager()

class PersonManager(models.Manager):
    def create_person(self, identifier):
        person = self.create(identifier = identifier)
        return person

但我不知道在哪里检查和获取现有的 person 对象。

【问题讨论】:

【参考方案1】:

目前尚不清楚您的问题是询问get_or_create 方法(至少可从Django 1.3 获得)还是update_or_create 方法(Django 1.7 中的新方法)。这取决于您要如何更新用户对象。

使用示例如下:

# In both cases, the call will get a person object with matching
# identifier or create one if none exists; if a person is created,
# it will be created with name equal to the value in `name`.

# In this case, if the Person already exists, its existing name is preserved
person, created = Person.objects.get_or_create(
        identifier=identifier, defaults="name": name
)

# In this case, if the Person already exists, its name is updated
person, created = Person.objects.update_or_create(
        identifier=identifier, defaults="name": name
)

【讨论】:

+1 for update_or_create 因为对于那个用例,它比 get_or_create 更好,只是在对象上再次调用 save() 在我的例子中,代替标识符,我有一组字段,它们形成了 unique_together。本案使用情况如何? @OmkarDeshpande 标识符只是一个例子;您可以传入任何有效的 django 查询,例如 Person.objects.get_or_create(a=a, b=b, c=c, defaults="name": name) 如果我想向创建的 Person 添加比默认字段更多的字段怎么办? @LoBellin defaults 这些函数的参数是您指定要添加到模型中的字段的位置。它与模型字段上指定的默认值无关【参考方案2】:

如果您正在寻找“如果存在则更新”用例,请参考@Zags excellent answer


Django 已经有一个get_or_create、https://docs.djangoproject.com/en/dev/ref/models/querysets/#get-or-create

对你来说可能是:

id = 'some identifier'
person, created = Person.objects.get_or_create(identifier=id)

if created:
   # means you have created a new person
else:
   # person just refers to the existing one

【讨论】:

【参考方案3】:

Django 对此有支持,请查看get_or_create

person, created = Person.objects.get_or_create(name='abc')
if created:
    # A new person object created
else:
    # person object already exists

【讨论】:

【参考方案4】:

update_or_create 仅适用于少量对象,但如果您要处理大型集合,则无法很好地扩展。 update_or_create 总是先运行 SELECT,然后再运行 UPDATE。

for the_bar in bars:
    updated_rows = SomeModel.objects.filter(bar=the_bar).update(foo=100)
        if not updated_rows:
            # if not exists, create new
            SomeModel.objects.create(bar=the_bar, foo=100)

这充其量只会运行第一个更新查询,并且只有当它匹配零行时才会运行另一个 INSERT 查询。如果您希望大多数行实际存在,这将大大提高您的性能。

不过,这一切都取决于您的用例。如果您期望主要是插入,那么bulk_create() 命令可能是一个选项。

【讨论】:

【参考方案5】:

我想我会添加一个答案,因为您的问题标题看起来像是在询问如何创建或更新,而不是按照问题正文中的描述获取或创建。

如果您确实想要创建或更新一个对象,默认情况下 .save() 方法已经具有此行为,来自the docs:

Django 抽象出使用 INSERT 或 UPDATE SQL 语句的需要。 具体来说,当你调用 save() 时,Django 遵循这个算法:

如果对象的主键属性设置为计算值 为 True(即,除 None 或空字符串以外的值),Django 执行更新。如果对象的主键属性没有设置 或者如果 UPDATE 没有更新任何东西,Django 会执行 INSERT。

值得注意的是,当他们说“如果 UPDATE 没有更新任何内容”时,他们实质上是指您提供给对象的 id 不存在于数据库中的情况。

【讨论】:

【参考方案6】:

您也可以像 get_or_create 一样使用 update_or_create,下面是我为 update_or_create 遵循的模式,假设模型 Person 具有 id(键)、名称、年龄、is_manager 作为属性 -

update_values = "is_manager": False
new_values = "name": "Bob", "age": 25, "is_manager":True

obj, created = Person.objects.update_or_create(identifier='id',
                                               defaults=update_values)
if created:
    obj.update(**new_values)

【讨论】:

【参考方案7】:

如果您创建时的输入之一是主键,这就足够了:

Person.objects.get_or_create(id=1)

如果存在两个具有相同主键的数据是不允许的,它将自动更新。

【讨论】:

【参考方案8】:

这应该是您正在寻找的答案

EmployeeInfo.objects.update_or_create(
    #id or any primary key:value to search for
    identifier=your_id, 
    #if found update with the following or save/create if not found
    defaults='name':'your_name'
)

【讨论】:

以上是关于创建 Django 模型或更新(如果存在)的主要内容,如果未能解决你的问题,请参考以下文章

django 模型创建连接到 mssql

如何以 django 模型形式覆盖保存方法

多对多查询的 Django 模型文件更新

如何访问 django settings.py 中的模型或视图属性?

如何根据具有外键关系的另一个模型更新 Django 模型的字段

如何使用 Django Rest Framework 创建多个模型实例?