鍍金池/ 問(wèn)答/Python  數(shù)據(jù)庫(kù)/ MongoDB 多個(gè)內(nèi)嵌文檔同時(shí)更新的問(wèn)題

MongoDB 多個(gè)內(nèi)嵌文檔同時(shí)更新的問(wèn)題

我正在使用MongoDB作為數(shù)據(jù)庫(kù),今天在試圖同時(shí)更新一個(gè)文檔的多個(gè)內(nèi)嵌文檔時(shí)遇到了問(wèn)題。

使用環(huán)境:Python3.6 PyMongo驅(qū)動(dòng)

在更新之前的表格式:

{
    "_id" : 0,
    "sys" : {
        "group" : 0,
        "status" : 1,
        "regtime" : NumberLong(1527240646218)
    },
    "info" : {
        "name" : "桂小方"
    }
}

已經(jīng)生成的更新數(shù)據(jù):

{
    'sys': {
        'group': 0,
        'status': 2
    },
    'info': {
        'name': '桂小方',
        'mail': 'abc@abc.com',
        'phone': '+1234567'
    }
}

我需要得到的更新之后的表:

{
    "_id" : 0,
    "sys" : {
        "group" : 0,
        "status" : 2, // 這里發(fā)生更新
        "regtime" : NumberLong(1527240646218)
    },
    "info" : {
        "name" : "桂小方",
        'mail': 'abc@abc.com', // 這里發(fā)生更新
        'phone': '+1234567' // 這里發(fā)生更新
    }
}

但當(dāng)我試圖使用 $set 更新器在以下語(yǔ)句中更新文檔:

database.users.update(
    {"_id": uid},
    {"$set": data} //這里的data代指上面的更新數(shù)據(jù)
)

但是卻得到了這樣的結(jié)果:

{
    "_id" : 0,
    "sys" : {
        "group" : 0,
        "status" : 2 // 我需要保留的regtime鍵消失了
    },
    "info" : {
        "name" : "桂小方",
        'mail': 'abc@abc.com', // 這里成功更新
        'phone': '+1234567' // 這里成功更新
    }
}

在Google上搜索之后我明白了問(wèn)題的原因:使用 $set 更新器會(huì)對(duì)文檔中指定的鍵進(jìn)行值替換,但是內(nèi)嵌文檔同樣會(huì)整個(gè)全部被提換成新的文檔;如果需要部分更新內(nèi)嵌文檔的值需要進(jìn)行 info.$ 這樣的更新。

但是我的表沒(méi)有固定的模式,更新數(shù)據(jù)也是生成的,所以想請(qǐng)問(wèn)大家有沒(méi)有直接可以同時(shí)對(duì)多個(gè)內(nèi)嵌文檔進(jìn)行更新的方法。謝謝!

回答
編輯回答
墻頭草

你的理解可能有誤。info.$是對(duì)匹配的數(shù)組元素的更新,跟內(nèi)嵌文檔沒(méi)有什么關(guān)系。
不知道你上面的data具體內(nèi)容到底是什么。從結(jié)果來(lái)看應(yīng)該是:

data = {
    "sys" : {
        "group" : 0,
        "status" : 2
    },
    "info" : {
        "name" : "桂小方",
        'mail': 'abc@abc.com', // 這里發(fā)生更新
        'phone': '+1234567' // 這里發(fā)生更新
    }
}

才有可能得到你的結(jié)果。如果data確實(shí)是這個(gè)值,那這個(gè)結(jié)果是沒(méi)有問(wèn)題的。想得到你要的結(jié)果,正確的更新語(yǔ)句應(yīng)該是:

db.users.update({
    "_id": uid
}, {
    'sys.status': data.sys.status,
    'info': data.info
});
2017年7月10日 02:35