侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

mongodb 修改器($inc/$set/$unset/$push/$pop/upsert)

数据库  /  管理员 发布于 5年前   192

对于文档的更新除替换外,针对某个或多个文档只需要部分更新可使用原子的更新修改器,能够高效的进行文档更新。更新修改器是中特殊的键,
用来指定复杂的操作,比如增加、删除或者调整键,还可能是操作数组或者内嵌文档。

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 = 5555> db.a.save(o)> db.a.find(){ "_id" : ObjectId("50026affdeb4fa8d154f8572"), "desc" : "hello world1!", "num": 55, "sname" : "jk", "type" : "1", "uid" : "20120002" }


  • 上一条:
    mongodb 数据类型(null/字符串/数字/日期/内嵌文档/数组等)
    下一条:
    MongoDB插入、更新、删除文档实现代码
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 分库分表的目的、优缺点及具体实现方式介绍(0个评论)
    • DevDB - 在 VS 代码中直接访问数据库(0个评论)
    • 在ubuntu系统中实现mysql数据存储目录迁移流程步骤(0个评论)
    • 在mysql中使用存储过程批量新增测试数据流程步骤(0个评论)
    • php+mysql数据库批量根据条件快速更新、连表更新sql实现(0个评论)
    • 近期文章
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2017-06
    • 2017-08
    • 2017-09
    • 2017-10
    • 2017-11
    • 2018-01
    • 2018-05
    • 2018-10
    • 2018-11
    • 2020-02
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2021-02
    • 2021-04
    • 2021-07
    • 2021-08
    • 2021-11
    • 2021-12
    • 2022-02
    • 2022-03
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-07
    • 2023-08
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-03
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客