2010-12-07 13 views
91

データベースからDjangoオブジェクトの状態をリフレッシュすることはできますか?データベースからdjangoオブジェクトをリロードする

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

UPD:トラッカーで再度開く/ WONTFIX戦争見つかり:http://code.djangoproject.com/ticket/901私はとほぼ同等の挙動を意味します。 メンテナがこれを気に入らない理由はまだ分かりません。

+0

通常のSQLコンテキストでは、これは意味をなさない。データベースオブジェクトはトランザクションが終了した後にのみ変更することができ、 'コミットする 'ことができます。これを済ませたら、次のSQLトランザクションがコミットするのを待つ必要があります。それはなぜですか?どのくらい次の取引を待つつもりですか? –

+0

これは不必要な機能のようです。データベースからオブジェクトをただちに再ルックアップすることはすでに可能です。 – Stephan

+0

私はこれも好きですが、繰り返しここでシャットダウンされています(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

答えて

134

オブジェクトをリフレッシュすると、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ええ、それは1.8でのみ実装されました。以前のバージョンのDjangoでは、他の答えの中で一番うまくいきます。 –

+0

ドキュメントに「すべての非遅延フィールドが更新されました」と記載されていますか? – Yunti

25

私はそうのようなreload the object from the databaseに、それは比較的簡単に見つけた:

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

はい、しかし...その後、このオブジェクトへのすべての参照を更新する必要があります。あまり便利でエラーが起こりにくい – grep

+0

はい、最後にform.instance.save()を呼び出すモデルが "reloaded"バージョンを取得するように、インスタンスが引き続き渡されている場合(ModelFormのサブクラスなど)は何もしません。 – eruciform

+2

これが必要であることがわかりましたCeleryがdjangoの外のdbでオブジェクトを更新したとき、djangoはオブジェクトが変更されたことを知らないのでオブジェクトのキャッシュを保持していたようです。 –

13

を@にはgrepのコメントを参照するには、それを行うのは可能であるべきではない:Djangoの1.8のよう

# 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

解決に感謝します。もしSOだけが複数のアップ投票を許せば! – user590028

+2

Djangoは 'refresh_from_db'メソッドを提供するようになりました。 – Flimm

関連する問題