mongoDB对于文档的更新,既可以实现文档的更新,也可以实现文档随意的增减键(列),这是与传统的关系型数据库最大的不同之处,也就是所谓的无模式带来的一定程度上的便利。即mongoDB支持文档更新,也支持文档替换。本文给出了mongoDB更新语法及示例。
一、文档更新语法
db.collection.update(
<query>, //查询或过滤条件
<update>, //修改器(被修改键及内容)
{
upsert: <boolean>, //为true或者false,如果为true,未找到匹配文档则创建新文档
multi: <boolean>, //用于确定是单行还是更新所有行(true为所有行)
writeConcern: <document> //设定写关注,用于确保强一致性还是弱一致性
} //后面的3.2之后的语法参数基本相同
)
db.collection.updateOne( //仅3.2之后版本
<filter>,
<update>,
{
upsert: <boolean>,
writeConcern: <document>
}
)
db.collection.updateMany( //仅3.2之后版本
<filter>,
<update>,
{
upsert: <boolean>,
writeConcern: <document>
}
)
db.collection.replaceOne( //仅3.2之后版本,用于替换文档
<filter>,
<replacement>,
{
upsert: <boolean>,
writeConcern: <document>
}
)
更新相关事项
mongoDB中所有写操作是基于单个文档基本的原子性操作(3.2版本后)
_id类不支持更新,也不支持使用一个不同的_id文档进行替换
文档的更新可能会导致文档变大,会重新申请及分配新的磁盘空间
二、演示文档更新
1、db.collection.updateOne()
//演示环境
db.version()
3.2.9
//插入演示数据,此处略,参考:mongoDB文档查询 插入演示数据部分
//链接地址
//下面查询favorites.artist值为Picasso的文档
> db.users.find({"favorites.artist": "Picasso"},
{finished:0,points:0,badges:0}).pretty()
{
"_id" : 1,
"name" : "sue",
"age" : 19,
"type" : 1,
"status" : "P",
"favorites" : {
"artist" : "Picasso",
"food" : "pizza"
}
}
{
"_id" : 6,
"name" : "abc",
"age" : 43,
"type" : 1,
"status" : "A",
"favorites" : {
"food" : "pizza",
"artist" : "Picasso"
}
}
//下面对favorites.artist值为Picasso的文档进行更新
//使用$set修改器来设置某个列的值,如下设置favorites.food的值pie,且type值为3
//使用$currentDate为新增的列lastModified设置当前日期
> db.users.updateOne(
{ "favorites.artist": "Picasso" },
{
$set: { "favorites.food": "pie", type: 3 },
$currentDate: { lastModified: true }
}
)
{ "acknowledged" : true, "matchedCount" : 1, "modifiedCount" : 1 }
//查看更改后的文档结果
> db.users.find({"favorites.artist": "Picasso"},
{finished:0,points:0,badges:0}).pretty()
{
"_id" : 1, //这个文档已经被修改,且新增了列
"name" : "sue",
"age" : 19,
"type" : 3,
"status" : "P",
"favorites" : {
"artist" : "Picasso",
"food" : "pie"
},
"lastModified" : ISODate("2016-09-30T07:25:28.135Z")
}
{
"_id" : 6, //而_id为6的这个文档没有被修改,即updateOne方法仅更新一个文档
"name" : "abc",
"age" : 43,
"type" : 1,
"status" : "A",
"favorites" : {
"food" : "pizza",
"artist" : "Picasso"
}
}
2、db.collection.updateMany()
//下面更新favorites.artist值为Picasso的文档,
//更新的内容为favorites.artist值为Pisanello,type的值为3,且增加或修改lastModified字段
> db.users.updateMany(
{ "favorites.artist": "Picasso" },
{
$set: { "favorites.artist": "Pisanello", type: 3 },
$currentDate: { lastModified: true }
}
)
{ "acknowledged" : true, "matchedCount" : 2, "modifiedCount" : 2 } //此处提示有2个匹配的文档,更新数为2
//再次查询favorites.artist值为Picasso的文档,其结果无任何文档返回,因为已经被更新
> db.users.find({"favorites.artist": "Picasso"},
... {finished:0,points:0,badges:0}).pretty()
//查询favorites.artist值为Pisanello的文档
> db.users.find({"favorites.artist": "Pisanello"},
... {finished:0,points:0,badges:0}).pretty()
{
"_id" : 1,
"name" : "sue",
"age" : 19, //Author : Leshami
"type" : 3, //Blog :
"status" : "P",
"favorites" : {
"artist" : "Pisanello",
"food" : "pie"
},
"lastModified" : ISODate("2016-09-30T08:00:47.826Z")
}
{
"_id" : 6,
"name" : "abc",
"age" : 43,
"type" : 3,
"status" : "A",
"favorites" : {
"food" : "pizza",
"artist" : "Pisanello"
},
"lastModified" : ISODate("2016-09-30T08:00:47.826Z")
}
3、db.collection.update()
//下面使用db.collection.update()方式更新
> db.users.update(
... { "favorites.artist": "Pisanello" },
... {
... $set: { "favorites.food": "pizza", type: 0, },
... $currentDate: { lastModified: true }
... }
... ) //此更新结果,如下行,nMatched为一个文档,nModified为1个文档,即仅更新一个文档
WriteResult({ "nMatched" : 1, "nUpserted" : 0, "nModified" : 1 }) //等同于updateOne方法
> db.users.update(
... { "favorites.artist": "Pisanello" },
... {
... $set: { "favorites.food": "pizza", type: 0, },
... $currentDate: { lastModified: true }
... },
... { multi: true } //使用参数multi,且值为true
... ) //从返回的结果可知,更新文档数为2,即所有满足条件的文档都被更新
WriteResult({ "nMatched" : 2, "nUpserted" : 0, "nModified" : 2 }) //等同于updateMany方法
//验证更新之后的结果
> db.users.find({"favorites.artist": "Pisanello"},
... {finished:0,points:0,badges:0}).pretty()
{
"_id" : 1,
"name" : "sue",
"age" : 19,
"type" : 0,
"status" : "P",
"favorites" : {
"artist" : "Pisanello",
"food" : "pizza"
},
"lastModified" : ISODate("2016-09-30T09:31:10.524Z")
}
{
"_id" : 6,
"name" : "abc",
"age" : 43,
"type" : 0,
"status" : "A",
"favorites" : {
"food" : "pizza",
"artist" : "Pisanello"
},
"lastModified" : ISODate("2016-09-30T09:31:10.524Z")
}
4、db.collection.replaceOne()
//替换前文档的内容(name:"abc")
> db.users.find({name:"abc"},{finished:0,points:0,badges:0}).pretty()
{
"_id" : 6,
"name" : "abc",
"age" : 43,
"type" : 1,
"status" : "A",
"favorites" : {
"food" : "pizza",
"artist" : "Picasso"
}
}
//下面使用db.collection.replaceOne()替换用户名为abc的文档,且替换了很多不同的键值
> db.users.replaceOne(
... { name: "abc" },
... { name: "amy", age: 34, type: 2, status: "P", favorites: { "artist": "Dali", food: "donuts" } }
... )
{ "acknowledged" : true, "matchedCount" : 1, "modifiedCount" : 1 } //这里提示有一个匹配,一个被更新
//验证替换后的结果
> db.users.find({name:"amy"},{finished:0,points:0,badges:0}).pretty()
{
"_id" : 6,
"name" : "amy",
"age" : 34,
"type" : 2,
"status" : "P",
"favorites" : {
"artist" : "Dali",
"food" : "donuts"
}
}
//使用db.collection.update方式可以实现相同的效果
> db.users.update(
... { name: "xyz" },
... { name: "mee", age: 25, type: 1, status: "A", favorites: { "artist": "Matisse", food: "mango" } }
... )
WriteResult({ "nMatched" : 1, "nUpserted" : 0, "nModified" : 1 })
三、小结
1、mongoDB文档更新有很多个不同的方法,传统的update,以及3.2版本之后的updateOne,updateMany
2、mongoDB文档替换也有很多个不通的方法,传统的update,以及3.2版本之后的replaceOnye,replaceMany
3、updateOne与updateMany是对update方法的扩展,update方法可以通过multi值为true或false来等同于updateMany以及updateOne
4、replaceOne与replaceMany也是对update方法的扩展,update方法可以通过multi值为true或false来等同于replaceMany以及replaceOne