设为首页 收藏本站
查看: 1530|回复: 0

[经验分享] 【MongoDB】4.MongoDB 原子修改器的 极速修改

[复制链接]

尚未签到

发表于 2017-12-16 18:49:00 | 显示全部楼层 |阅读模式
  文档转自:http://blog.csdn.net/mcpang/article/details/7752736
  对于文档的更新除替换外,针对某个或多个文档只需要部分更新可使用原子的更新修改器,能够高效的进行文档更新。更新修改器是中特殊的键,
  用来指定复杂的操作,比如增加、删除或者调整键,还可能是操作数组或者内嵌文档。
  1.$inc
  --------------------------------------------------------------------------
  这个修改器干什么使的呢?看看下面示例的具体操作后的结果即可知道。
  示例文档:{"uid":"201203","type":"1",size:10}
  > db.b.insert({"uid":"201203","type":"1",size:10})
  > db.b.find()
  { "_id" : ObjectId("5003b6135af21ff428dafbe6"), "uid" : "201203", "type" : "1",
  "size" : 10 }
  > db.b.update({"uid" : "201203"},{"$inc":{"size" : 1}})
  > db.b.find()
  { "_id" : ObjectId("5003b6135af21ff428dafbe6"), "uid" : "201203", "type" : "1",
  "size" : 11 }
  > db.b.update({"uid" : "201203"},{"$inc":{"size" : 2}})
  > db.b.find()
  { "_id" : ObjectId("5003b6135af21ff428dafbe6"), "uid" : "201203", "type" : "1",
  "size" : 13 }
  > db.b.update({"uid" : "201203"},{"$inc":{"size" : -1}})
  > db.b.find()
  { "_id" : ObjectId("5003b6135af21ff428dafbe6"), "uid" : "201203", "type" : "1",
  "size" : 12 }
  得出结论:修改器$inc可以对文档的某个值为数字型(只能为满足要求的数字)的键进行增减的操作。
  (这里有个问题:上篇中说到更新默认只对满足条件的记录集中第一个文档进行更新,那么使用$inc修改器之后,还是一样吗?)
  2.$set
  -------------------------------------------------------------------
  用来指定一个键并更新键值,若键不存在并创建。来看看下面的效果:
  > db.a.findOne({"uid" : "20120002","type" : "3"})
  { "_id" : ObjectId("500216de81b954b6161a7d8f"), "desc" : "hello world2!", "num"
  : 40, "sname" : "jk", "type" : "3", "uid" : "20120002" }
  --size键不存在的场合
  > db.a.update({"uid" : "20120002","type" : "3"},{"$set":{"size":10}})
  > db.a.findOne({"uid" : "20120002","type" : "3"})
  { "_id" : ObjectId("500216de81b954b6161a7d8f"), "desc" : "hello world2!", "num"
  : 40, "size" : 10, "sname" : "jk", "type" : "3", "uid" : "20120002" }
  --sname键存在的场合
  > db.a.update({"uid" : "20120002","type" : "3"},{"$set":{"sname":"ssk"}})
  > db.a.find()
  { "_id" : ObjectId("500216de81b954b6161a7d8f"), "desc" : "hello world2!", "num"
  : 40, "size" : 10, "sname" : "ssk", "type" : "3", "uid" : "20120002" }
  { "_id" : ObjectId("50026affdeb4fa8d154f8572"), "desc" : "hello world1!", "num"
  : 50, "sname" : "jk", "type" : "1", "uid" : "20120002" }
  --可改变键的值类型
  > db.a.update({"uid" : "20120002","type" : "3"},{"$set":{"sname":["Java",".net","c++"]}})
  > db.a.findOne({"uid" : "20120002","type" : "3"})
  {
  "_id" : ObjectId("500216de81b954b6161a7d8f"),
  "desc" : "hello world2!",
  "num" : 40,
  "size" : 10,
  "sname" : [
  "java",
  ".net",
  "c++"
  ],
  "type" : "3",
  "uid" : "20120002"
  }
  对于内嵌的文档,$set又是如何进行更新的内嵌的文档的呢,请看下面的示例:
  示例文档:{"name":"toyota","type":"suv","size":{"height":10,"width":5,"length":15}}
  > db.c.findOne({"name":"toyota"})
  {
  "_id" : ObjectId("5003be465af21ff428dafbe7"),
  "name" : "toyota",
  "type" : "suv",
  "size" : {
  "height" : 10,
  "width" : 5,
  "length" : 15
  }
  }
  > db.c.update({"name":"toyota"},{"$set":{"size.height":8}})
  > db.c.findOne({"name":"toyota"})
  {
  "_id" : ObjectId("5003be465af21ff428dafbe7"),
  "name" : "toyota",
  "type" : "suv",
  "size" : {
  "height" : 8,
  "width" : 5,
  "length" : 15
  }
  }
  > db.c.update({"name":"toyota"},{"$set":{"size.width":7}})
  > db.c.findOne({"name":"toyota"})
  {
  "_id" : ObjectId("5003be465af21ff428dafbe7"),
  "name" : "toyota",
  "type" : "suv",
  "size" : {
  "height" : 8,
  "width" : 7,
  "length" : 15
  }
  }
  可见:对于内嵌文档在使用$set更新时,使用"."连接的方式。
  3.$unset
  ----------------------------------------------------------------
  从字面就可以看出其意义,主要是用来删除键。
  示例操作效果如下:
  > db.a.update({"uid" : "20120002","type" : "3"},{"$unset":{"sname":1}})
  > db.a.findOne({"uid" : "20120002","type" : "3"})
  {
  "_id" : ObjectId("500216de81b954b6161a7d8f"),
  "desc" : "hello world2!",
  "num" : 40,
  "size" : 10,
  "type" : "3",
  "uid" : "20120002"
  }
  > db.a.update({"uid" : "20120002","type" : "3"},{"$unset":{"num":0}})
  > db.a.findOne({"uid" : "20120002","type" : "3"})
  {
  "_id" : ObjectId("500216de81b954b6161a7d8f"),
  "desc" : "hello world2!",
  "size" : 10,
  "type" : "3",
  "uid" : "20120002"
  }
  > db.a.update({"uid" : "20120002","type" : "3"},{"$unset":{"size":-1}})
  > db.a.findOne({"uid" : "20120002","type" : "3"})
  {
  "_id" : ObjectId("500216de81b954b6161a7d8f"),
  "desc" : "hello world2!",
  "type" : "3",
  "uid" : "20120002"
  }
  > db.a.update({"uid" : "20120002","type" : "3"},{"$unset":{"desc":"sssssss"}})
  > db.a.findOne({"uid" : "20120002","type" : "3"})
  {
  "_id" : ObjectId("500216de81b954b6161a7d8f"),
  "type" : "3",
  "uid" : "20120002"
  }
  得出结论:使用修改器$unset时,不论对目标键使用1、0、-1或者具体的字符串等都是可以删除该目标键。
  4.数组修改器--$push
  ------------------------------------------------------------------
  示例操作效果如下:
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "type" : "suv",
  "size" : { "height" : 8, "width" : 7, "length" : 15 } }
  --先push一个当前文档中不存在的键title
  > db.c.update({"name" : "toyota"},{$push:{"title":"t1"}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1" ], "type" : "suv" }
  --再向title中push一个值
  > db.c.update({"name" : "toyota"},{$push:{"title":"t2"}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t2" ], "type" : "suv" }
  --再向title中push一个值
  > db.c.update({"name" : "toyota"},{$push:{"title":"t2"}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t2", "t2" ], "type" : "suv" }
  --再向一个已经存在的键值非数组类型的键push一个值
  > db.c.update({"name" : "toyota"},{$push:{"size.height":10}})
  Cannot apply $push/$pushAll modifier to non-array
  > db.c.update({"name" : "toyota"},{$push:{"name":"ddddddd"}})
  Cannot apply $push/$pushAll modifier to non-array
  得出结论:$push--向文档的某个数组类型的键添加一个数组元素,不过滤重复的数据。添加时键存在,要求键值类型必须是数组;键不存在,则创建数组类型的键。
  5.数组修改器--$ne/$addToSet
  ---------------------------------------------------------------------
  主要给数组类型键值添加一个元素时,避免在数组中产生重复数据,$ne在有些情况是不通行的。
  > db.c.update({"title" : {$ne:"t2"}},{$push:{"title":"t2"}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t2", "t2" ], "type" : "suv" }
  > db.c.update({"name" : "toyota"},{$addToSet:{"title":"t2"}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t2", "t2" ], "type" : "suv" }
  6.数组修改器--$pop、$pull
  ------------------------------------------------------------
  $pop从数组的头或者尾删除数组中的元素,示例如下:
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t2", "t3", "t4" ],"type" : "suv" }
  --从数组的尾部删除 1
  > db.c.update({"name" : "toyota"},{$pop:{"title":1}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t2", "t3" ], "type" : "suv" }
  --从数组的头部 -1
  > db.c.update({"name" : "toyota"},{$pop:{"title":-1}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t2", "t3" ], "type" : "suv" }
  --从数组的尾部删除 0
  > db.c.update({"name" : "toyota"},{$pop:{"title":0}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t2" ], "type" : "suv" }
  $pull从数组中删除满足条件的元素,示例如下:
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t2", "t2", "t3" ],"type" : "suv" }
  > db.c.update({"name" : "toyota"},{$pull:{"title":"t2"}})
  > db.c.find()
  { "_id" : ObjectId("5003be465af21ff428dafbe7"), "name" : "toyota", "size" : { "height" : 8,
  "width" : 7, "length" : 15 }, "title" : [ "t1", "t3" ], "type" : "suv" }
  7.数组的定位修改器
  -------------------------------------------------------------------
  在需要对数组中的值进行操作的时候,可通过位置或者定位操作符("$").数组是0开始的,可以直接将下标作为键来选择元素。
  示例如下:
  {"uid":"001",comments:[{"name":"t1","size":10},{"name":"t2","size":12}]}
  > db.c.find({"uid":"001"})
  { "_id" : ObjectId("5003da405af21ff428dafbe8"), "uid" : "001", "comments" : [ {
  "name" : "t1", "size" : 10 }, { "name" : "t2", "size" : 12 } ] }
  > db.c.update({"uid":"001"},{$inc:{"comments.0.size":1}})
  > db.c.find({"uid":"001"})
  { "_id" : ObjectId("5003da405af21ff428dafbe8"), "uid" : "001", "comments" : [ {
  "name" : "t1", "size" : 11 }, { "name" : "t2", "size" : 12 } ] }
  > db.c.update({"comments.name":"t1"},{$set:{"comments.$.size":1}})
  > db.c.find({"uid":"001"})
  { "_id" : ObjectId("5003da405af21ff428dafbe8"), "uid" : "001", "comments" : [ {
  "name" : "t1", "size" : 1 }, { "name" : "t2", "size" : 12 } ] }
  --若为多个文档满足条件,则只更新第一个文档。
  8.upsert
  -----------------------------------------------------------------
  upsert是一种特殊的更新。当没有符合条件的文档,就以这个条件和更新文档为基础创建一个新的文档,如果找到匹配的文档就正常的更新。
  使用upsert,既可以避免竞态问题,也可以减少代码量(update的第三个参数就表示这个upsert,参数为true时)
  > db.c.remove()
  > db.c.update({"size":11},{$inc:{"size":3}})
  > db.c.find()
  > db.c.update({"size":11},{$inc:{"size":3}},false)
  > db.c.find()
  > db.c.update({"size":11},{$inc:{"size":3}},true)
  > db.c.find()
  { "_id" : ObjectId("5003ded6c28f67507a6df1de"), "size" : 14 }
  9.save函数
  -----------------------------------------------------------------
  1.可以在文档不存在的时候插入,存在的时候更新,只有一个参数文档。
  2.要是文档含有"_id",会调用upsert。否则,会调用插入。
  > db.a.find()
  { "_id" : ObjectId("50026affdeb4fa8d154f8572"), "desc" : "hello world1!", "num": 50,
  "sname" : "jk", "type" : "1", "uid" : "20120002" }
  > var o = db.a.findOne()
  > o.num = 55
  55
  > db.a.save(o)
  > db.a.find()
  { "_id" : ObjectId("50026affdeb4fa8d154f8572"), "desc" : "hello world1!", "num": 55,
  "sname" : "jk", "type" : "1", "uid" : "20120002" }

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-424788-1-1.html 上篇帖子: Ubuntu14.04下Mongodb(离线安装方式 下篇帖子: 用Node.JS+MongoDB搭建个人博客(安装环境)(一)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表