2010-12-07 3 views
91

Est-il possible d'actualiser l'état d'un objet django à partir de la base de données? Je veux dire un comportement à peu près équivalent à:Recharger l'objet django à partir de la base de données

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

UPD: Trouvé une guerre rouvrir/wontfix dans le suivi: http://code.djangoproject.com/ticket/901. Je ne comprends toujours pas pourquoi les mainteneurs n'aiment pas ça.

+0

Dans un contexte SQL ordinaire, cela n'a pas de sens. L'objet de base de données ne peut être modifié * qu'après * votre transaction se termine et fait un 'commmit'. Une fois que vous avez fait cela, vous devrez attendre la prochaine transaction SQL à valider. Pourquoi faire ça? Combien de temps allez-vous attendre la prochaine transaction? –

+0

Cela semble être une fonction inutile; il est déjà possible de simplement rechercher de nouveau l'objet dans la base de données. – Stephan

+0

je voudrais aussi cela, mais il a été arrêté à plusieurs reprises [ici] (http://www.google.com/url?sa=t&source=web&cd=1&ved=0CBkQFjAA&url=https%3A%2F%2Fcode.djangoproject .com% 2Fticket% 2F901 & ei = xOP7TaaCNMv1gAen2pTeCw & usg = AFQjCNFs7kMieML6P8vlIeQplFJpVxQbTA) – eruciform

Répondre

134

A partir de Django 1.8 les objets rafraîchissants sont intégrés. Link to docs.

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's updated value needs to be reloaded 
    # from the database. 
    obj.refresh_from_db() 
    self.assertEqual(obj.val, 2) 
+0

@ fcracker79 Oui, il a seulement été implémenté en 1.8. Pour les versions antérieures de Django, il vaut mieux utiliser l'une des autres réponses. –

+0

Vous ne savez pas ce que "Tous les champs non différés sont mis à jour" mentionnés dans les documents? – Yunti

25

Je l'ai trouvé relativement facile à reload the object from the database comme ceci:

x = X.objects.get(id=x.id) 
+14

Oui, mais ... après cela, vous devez mettre à jour toutes les références à cet objet. Pas très pratique et sujet aux erreurs. – grep

+0

oui, cela ne fait rien si l'instance est toujours transmise, par exemple par une sous-classe de ModelForm, de sorte que celui qui appelle enfin form.instance.save() obtiendra la version "reloaded" – eruciform

+2

a trouvé cela nécessaire Lorsque Celery a mis à jour mon objet dans la base de données en dehors de django, django a apparemment gardé un cache de l'objet car il n'avait aucune idée qu'il avait changé. –

13

En ce qui concerne le commentaire de @ grep, il ne devrait pas être possible de le faire:

# Put this on your base model (or monkey patch it onto django's Model if that's 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

Merci pour la solution. Si seulement SO a permis plusieurs votes up! – user590028

+2

Django fournit maintenant la méthode 'refresh_from_db'. – Flimm

Questions connexes