MongoDB入门4——更新文档和修改器2

来源:未知 责任编辑:智问网络 发表时间:2013-10-22 19:28 点击:
c)数组修改器
 
 
    数组修改器,顾名思义,当然是操作数组的啦。一般来说多数组的操作有若干不同种类的,因此,MongoDB也准备了不同的数组修改器。我们会一一学习。
 
$push修改器
    $push修改器能够向指定的数组的末端插入一个新的元素。假设这个数组本身都不存在,那么先创建这个数组。现在假设我现在发表一篇博客,刚一发表,就有人发表评论了。这时候就需要往评论列表中加一个评论记录。
 
 
> db.blog.insert({"title":"A New Blog","content":"MongoDB tutorial..."});   
> db.blog.update({"title":"A New Blog"},{"$push":{"comments":{"name":"xiaobai","content":"very good"}}});   
> db.blog.find();   
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [ { "name" : "xiaobai", "content" : "very good" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }   
>   
 
> db.blog.insert({"title":"A New Blog","content":"MongoDB tutorial..."});
> db.blog.update({"title":"A New Blog"},{"$push":{"comments":{"name":"xiaobai","content":"very good"}}});
> db.blog.find();
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [ { "name" : "xiaobai", "content" : "very good" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }
>  
 
    之前没记录,但是现在已经添加了一个评论数组,并且已经往里面加入了xiaobai的评论。现在再往里面添加一条评论:
 
 
> db.blog.update({"title":"A New Blog"},{"$push":{"comments":{"name":"xiaohong","content":"very very good"}}});   
> db.blog.find();   
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [     {     "name" : "xiaobai",     "content" : "very good" },     {    "name" : "xiaohong",     "content" : "very very good" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }   

>  <SPAN style="BACKGROUND-COLOR: #ffffff; FONT-FAMILY: Monaco; WHITE-SPACE: normal; COLOR: #555555; FONT-SIZE: 14px"> </SPAN> 
 
> db.blog.update({"title":"A New Blog"},{"$push":{"comments":{"name":"xiaohong","content":"very very good"}}});
> db.blog.find();
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [     {     "name" : "xiaobai",     "content" : "very good" },     {    "name" : "xiaohong",     "content" : "very very good" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }
>   
 
 
 
$ne修改器
    如果有这种情况:一个人只允许只允许评论一次,那么我们在添加新的评论时,就需要判断要添加的评论是否已经存在了。$ne修改器就是判断是否不存在某条记录,不存在的时候才执行某个操作。接着上面的例子,我们先查看,评论中没有lisi的评论的话,我们就插入一条新的lisi的评论:
 
 
> db.blog.update({"title":"A New Blog","comments.name":{"$ne":"lisi"}},{"$push":{"comments":{"name":"lisi","comment":"Not Bad"}}});   
> db.blog.find();   
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [     {     "name" : "xiaobai",     "content" : "very good" },     {    "name" : "xiaohong",     "content" : "very very good" },     {     "name" : "lisi",     "comment" : "Not Bad" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }   
>   
 
> db.blog.update({"title":"A New Blog","comments.name":{"$ne":"lisi"}},{"$push":{"comments":{"name":"lisi","comment":"Not Bad"}}});
> db.blog.find();
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [     {     "name" : "xiaobai",     "content" : "very good" },     {    "name" : "xiaohong",     "content" : "very very good" },     {     "name" : "lisi",     "comment" : "Not Bad" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }
>  现在,如果我们再次执行上面的update代码,就不会再添加lisi的评论了。这里我们就不再演示了。
 
$addToSet修改器
    这个修改器其实做的就是上面一样的事情。它会自动的判断某个键是否存在重复的值。不存在才添加记录。现在如果有一个关于用户信息的文档如下:
 
 
> var user = {"name":"Tom","emailaddrs":["a@b.com","b@b.com"]};   
> db.users.insert(user);   
> db.users.find();   
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "name" : "Tom", "emailaddrs" : [ "a@b.com", "b@b.com" ] }   
>   
 
> var user = {"name":"Tom","emailaddrs":["a@b.com","b@b.com"]};
> db.users.insert(user);
> db.users.find();
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "name" : "Tom", "emailaddrs" : [ "a@b.com", "b@b.com" ] }
>       现在我们需要再加一个邮件地址:c@b.com,我们可以使用这个修改器添加新的邮件地址:
 
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$addToSet":{"emailaddrs":"c@b.com"}});   
> db.users.find();   
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "a@b.com", "b@b.com", "c@b.com" ], "name" : "Tom" }   
>    
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$addToSet":{"emailaddrs":"c@b.com"}});
> db.users.find();
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "a@b.com", "b@b.com", "c@b.com" ], "name" : "Tom" }
>    再次执行上面的操作是不会重复添加邮件地址的。
 
$each修改器(配合$addToSet修改器使用)
    $addToSet配合$each修改器可以实现一次批量修改。接着上面邮件列表的例子,如果一次需要添加几个邮件地址,使用$ne和$push就不能一次直接添加多个地址。操作如下:
 
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$addToSet":{"emailaddrs":{"$each":["d@b.com","e@b.com"]}}});   
> db.users.find();   
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "a@b.com", "b@b.com", "c@b.com", "d@b.com", "e@b.com" ], "name" : "Tom" }   
>   
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$addToSet":{"emailaddrs":{"$each":["d@b.com","e@b.com"]}}});
> db.users.find();
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "a@b.com", "b@b.com", "c@b.com", "d@b.com", "e@b.com" ], "name" : "Tom" }
>   $pop修改器
 
    这个修改器可以将数组任何一端的第一个值删除。{"pop":{key:1}}将数组尾部值删掉,{"pop":{key:-1}}:将数组头部值删除。演示一下:
 
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$pop":{"emailaddrs":1}});   
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$pop":{"emailaddrs":-1}});   
> db.users.find();   
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "b@b.com", "c@b.com", "d@b.com" ], "name" : "Tom" }   
>   
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$pop":{"emailaddrs":1}});
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$pop":{"emailaddrs":-1}});
> db.users.find();
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "b@b.com", "c@b.com", "d@b.com" ], "name" : "Tom" }
>       这样,第一个和最后一个邮件地址都被删除了。
 
$pull修改器
    这个修改器也是用来删除数组中的元素的,但是是通过条件匹配来删除所有条件满足的值,而不仅仅是删某一个。接上面的例子,我们可以用它删除c@b.com
 
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$pull":{"emailaddrs":"c@b.com"}});   
> db.users.find();   
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "b@b.com", "d@b.com" ], "name" : "Tom" }   
>   
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$pull":{"emailaddrs":"c@b.com"}});
> db.users.find();
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "b@b.com", "d@b.com" ], "name" : "Tom" }
>      数组定位修改器
 
    数组定位可以通过两种方式实现:下标和$(定位操作符)。下标是从0开始的。
 
    数组定位方式修改第一个邮件地址:
 
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$set":{"emailaddrs.0":"bbb@b.com"}});   
> db.users.find();   
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "bbb@b.com", "d@b.com" ], "name" : "Tom" }   
>   
 
> db.users.update({"_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62")},{"$set":{"emailaddrs.0":"bbb@b.com"}});
> db.users.find();
{ "_id" : ObjectId("4ee49adeaf9c9fecc9c9cf62"), "emailaddrs" : [ "bbb@b.com", "d@b.com" ], "name" : "Tom" }
>       上面的情况有个不足:不查询出结果怎么知道顺序呢。为了解决这个问题,引入了$定位操作符。用来匹配查询条件查询出来的结果(也就是update的第一个参数)。
 
    回到上面那个评论的例子,我们要修改把评论人为xiaobai的评论的评论人改成xiaoxiaobai。
 
 
> db.blog.update({"comments.name":"xiaobai"},{"$set":{"comments.$.name":"xiaoxiaobai"}});   
> db.blog.find()   
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [     {     "content" : "very good",     "name" : "xiaoxiaobai" },    {     "name" : "xiaohong",     "content" : "very very good" },     {     "name" : "lisi",     "comment" : "Not Bad" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }   
>   
 
> db.blog.update({"comments.name":"xiaobai"},{"$set":{"comments.$.name":"xiaoxiaobai"}});
> db.blog.find()
{ "_id" : ObjectId("4ee47369af9c9fecc9c9cf61"), "comments" : [     {     "content" : "very good",     "name" : "xiaoxiaobai" },    {     "name" : "xiaohong",     "content" : "very very good" },     {     "name" : "lisi",     "comment" : "Not Bad" } ], "content" : "MongoDB tutorial...", "title" : "A New Blog" }
>       如果有多条匹配,则只会修改第一条。
 
     上面就是基本的修改器了。
 
     参考书籍:
 
    《MongoDB:The definitive guide》
 

摘自 wawlian说
    发表评论
    请自觉遵守互联网相关的政策法规,严禁发布色情、暴力、反动的言论。
    用户名: 验证码:点击我更换图片
    最新评论 更多>>

    推荐热点

    • Request.ServerVariables 参数大全
    • 执行全文索引时出现权限不足的解决方法
    • 导入excel文件处理流程节点的解决方案
    • 查看sql修改痕迹(SQL Change Tracking on Table)
    • MongoDB安装为Windows服务方法与注意事项
    • App数据层设计及云存储使用指南
    • PostgreSQL启动过程中的那些事三:加载GUC参数
    • 写给MongoDB开发者的50条建议Tip1
    • Percolator与分布式事务思考(二)
    网站首页 - 友情链接 - 网站地图 - TAG标签 - RSS订阅 - 内容搜索
    Copyright © 2008-2015 计算机技术学习交流网. 版权所有

    豫ICP备11007008号-1