Перезагрузить объект django из базы данных

83

Возможно ли обновить состояние объекта django из базы данных? Я имею в виду поведение, примерно эквивалентное:

new_self = self.__class__.objects.get(pk=self.pk)
for each field of the record:
    setattr(self, field, getattr(new_self, field))

UPD: Найден войну reopen/wontfix в трекере: http://code.djangoproject.com/ticket/901. Все еще не понимаю, почему сопровождающим это не нравится.

  • 0
    В обычном контексте SQL это не имеет смысла. Объект базы данных может быть изменен только после того, как ваша транзакция commmit и сделает commmit . Как только вы это сделаете, вам придется подождать, пока следующая транзакция SQL будет зафиксирована. Зачем это делать? Как долго вы собираетесь ждать следующей транзакции?
  • 0
    Это кажется ненужной функцией; уже возможно просто повторно найти объект из базы данных.
Показать ещё 2 комментария
Теги:
django-models

3 ответа

127
Лучший ответ

По состоянию на Django 1.8 встроены восстановительные объекты. Ссылка на документы.

def test_update_result(self):
    obj = MyModel.objects.create(val=1)
    MyModel.objects.filter(pk=obj.pk).update(val=F('val') + 1)
    # At this point obj.val is still 1, but the value in the database
    # was updated to 2. The object updated value needs to be reloaded
    # from the database.
    obj.refresh_from_db()
    self.assertEqual(obj.val, 2)
  • 0
    @ fcracker79 Да, это было реализовано только в версии 1.8. Для более ранних версий Django лучше всего использовать один из других ответов.
  • 1
    Не уверены, что означает "Все неотложенные поля обновлены", упомянутые в документе?
Показать ещё 1 комментарий
22

Я нашел относительно легким перезагрузить объект из базы данных так:

x = X.objects.get(id=x.id)
  • 14
    Да, но ... после этого вы должны обновить все ссылки на этот объект. Не очень удобно и подвержено ошибкам.
  • 0
    да, это ничего не делает, если экземпляр все еще передается, например, подклассом ModelForm, так что тот, который в конечном итоге вызывает form.instance.save (), получит «перезагруженную» версию
Показать ещё 6 комментариев
13

В отношении комментария @grep не следует делать:

# Put this on your base model (or monkey patch it onto django Model if that your thing)
def reload(self):
    new_self = self.__class__.objects.get(pk=self.pk)
    # You may want to clear out the old dict first or perform a selective merge
    self.__dict__.update(new_self.__dict__)

# Use it like this
bar.foo = foo
assert bar.foo.pk is None
foo.save()
foo.reload()
assert bar.foo is foo and bar.foo.pk is not None
  • 0
    Спасибо за решение. Если бы только ТАК допускалось несколько голосов "за"!
  • 8
    Django теперь предоставляет метод refresh_from_db .

Ещё вопросы

Сообщество Overcoder
Наверх
Меню