首頁 >後端開發 >Python教學 >如何使用 Django 的 ORM 使用 .update() 方法更新 JsonField 中多個鍵的值?

如何使用 Django 的 ORM 使用 .update() 方法更新 JsonField 中多個鍵的值?

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB轉載
2024-02-06 09:48:121089瀏覽

如何使用 Django 的 ORM 使用 .update() 方法更新 JsonField 中多个键的值?

問題內容

我有一個模型字段,我正在嘗試使用 django 的 .update() 方法進行更新。該字段是一個 jsonfield。我通常會在這個領域保存一本字典。例如{"test": "是", "prod": "否"}

這是模型:

class question(models.model):
    # {"test": "yes", "prod": "no"}
    extra_data = models.jsonfield(default=dict, null=true, blank=true)

我可以使用此查詢更新字典內的鍵(順便說一句,效果很好):

Question.filter(id="123").update(meta_data=Func(
                        F("extra_data"),
                        Value(["test"]),
                        Value("no", JSONField()),
                        function="jsonb_set",
                    ))

現在的問題是,有沒有辦法可以使用 .update() 一次更新多個金鑰(在我自己的情況下是兩個),如上面的查詢所示?

我想使用 .update() 方法而不是 .save() 來實現此目的,這樣我就可以避免呼叫 post_save 訊號函數。


正確答案


免責聲明:它看起來會很難看。

我過去曾這樣做過,儘管是使用純 sql,而不是使用 django。這個想法是遞歸呼叫jsonb_set()。每次呼叫都會設定一個密鑰。

因此,在 sql 中,要設定鍵 {"test":"yes","prod":"no"} 你可以這樣做:

update question set meta_data = jsonb_set(jsonb_set(extra_data, '{test}', '"yes"'::jsonb), '{prod}', '"no"'::jsonb)
where id = 123;

請注意,jsonb_set有兩個巢狀調用,最裡面的使用meta_data,最外面的接收最裡面的結果。

所以 django 的等價物看起來像這樣(注意,未經測試):

question.filter(id="123").update(
    meta_data=func(
        func(
            f("extra_data"),
            value(["test"]),
            value("yes", jsonfield()),
            function="jsonb_set",
        ),
        value(["prod"]),
        value("no", jsonfield()),
        function="jsonb_set",
    ))

或您可以建立一個遞歸函數,為您傳回混亂的情況,一次設定一個:

def recursive_jsonb_set(original, **kwargs):
    if kwargs:
        key, value = kwargs.popitem()
        return Func(
            recursive_jsonb_set(original, **kwargs),
            Value(key.split('__')),
            Value(value, JSONField()),
            function="jsonb_set")
    else:
        return original

Question.filter(id="123").update(
    meta_data=recursive_jsonb_set(F("extra_data"), test="yes", prod="no", other__status="done"))

請注意 '__' 功能以更新子項目。

以上是如何使用 Django 的 ORM 使用 .update() 方法更新 JsonField 中多個鍵的值?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:stackoverflow.com。如有侵權,請聯絡admin@php.cn刪除