mongodb 子文档查询
来源:互联网 发布:网络诈骗去哪里举报 编辑:程序博客网 时间:2024/05/17 23:00
mongodb 子文档查询 此博文包含图片 (2012-12-10 18:03:22)转载▼
标签: it 分类: mongoDB
首先我的数据库里有两个文档,每个文档里都有2个嵌套的数组:mongodb <wbr>子文档查询
如果我想查询comments里score大于5的记录:
testProvider.find({"comments.score":{"$gt":5}},{},function(err, result){
console.log(result);
});
返回了查找到的记录,说明了查找有效果了:
mongodb <wbr>子文档查询
这样查找返回的数据都是把匹配的整个文档返回过来了。
如果我觉得其中有个字段是没必要返回给我的可以设置这个字段为0:
testProvider.find({"comments.score":{"$gt":5}},{'comments.others':0},function(err, result) {
console.log(JSON.stringify(result));
});
可以看到返回的结果中已经没有others这个字段了
mongodb <wbr>子文档查询
如果我想查询comments数组里面的某个子文档中others数组里面name为a的记录:
testProvider.find({"comments.others.name":'a'},{},function(err, result) {
console.log(JSON.stringify(result));
});
返回的结果中的一部分如下图:
mongodb <wbr>子文档查询
从上面的几个示例中可以看到我想要查找某个符合条件的子文档时可以直接一直用'.'号表示下去而不用管其中是否有数组,如:{"comments.others.name":'a'}
一些高级的查询语法看这一篇:http://blog.sina.com.cn/s/blog_abba9c1d0101eti7.html
1、使用"$size"可以查询指定长度的数组
查询数组长度为3的数组db.users.find({"emails":{"$size":3}})
常见的查询是数组长度范围的查询."$size"并不能与其他查询子句组合(如:"$gt"),但是这种查询可以通过
在文档中添加一个"size"键的方式来实现.这样每一次向指定数组添加元素的时候,同时增加"size"值.原来这样
的更新: db.users.update({"$push":{"emails":"295240648@139.com"}})
变成这样的更新: db.users.update({"$push":{"emails":"295240648@139.com"},"$inc":{"size":1}})
这样就可以这样查询了 db.users.find({"size":{"$gt":3}})
2、使用"$slice"查询
find的第二个参数是可选的,可以指定返回那些键,"$slice"返回数组的一个子集合
返回emails数组的前两个元素 db.users.find({"userName":"refactor"},{"emails":{"$slice":2}})
返回emails数组的后两个元素 db.users.find({"userName":"refactor"},{"emails":{"$slice":-2}})
返回emails数组的第2个和第11个元素.如果数组不够11个,则返回第2个后面的所有元素
db.users.find({"userName":"refactor"},{"emails":{"$slice":[1,10]}})
"$slice"默认将返回文档中的所有键.
3、要正确指定一组条件,而不用指定每个键,要使用"$elemMatch".这种模糊的命名条件能用来部分指定匹配数组中
的单个内嵌文档的限定条件.正确写法应该是:
db.blog.find(
{
"comments":
{
"$elemMatch":
{
"author":"refactor",
"score":{"$gte":5}
}
}
}
)
"$elemMatch"将限定条件进行分组,仅当需要对一个内嵌文档的多个键操作时才会用到.
4."$where"查询
"$where"可以执行任意javascript作为查询的一部分.这使得查询能做(几乎)任何事情.
最典型的应用就是比较文档中的两个键的值是否相等.
如:db.blog.insert({"title":"refactor","content":"refactor"})
db.blog.insert({"title":"refactor1","content":"refactor content"})
第一个文档title键和content键的值相同.要返回该文档.
db.blog.find(
{
"$where":function()
{
for(var current in this)
{
for(var other in this)
{
if(current!=other&&this[current]==this[other])
{
return true;
}
}
}
return false;
}
}
)
如果函数返回true,文档就作为结果的一部分被返回.
上面是用一个函数,也可以用一个字符串指定"$where"查询.下面两种方式是等价的:
db.blog.find({"$where":"this.x+this.y==10"})
db.blog.find({"$where":"function(){return this.x+this.y==10;}"})
不是非常必要时,应避免"$where"查询.它在速度上要比常规查询慢的多.每个文档都要从BSON转换成
javascript对象,然后通过"$where"的表达式来运行.它还不能利用索引.
将常规查询作为前置过滤,与"$where"组合使用才能不牺牲性能,如果有可能的话,用索引根据非"$where"
子句进行过滤,"$where"只用于对结果进行调优
上面的4个查询转自:http://www.cnblogs.com/refactor/archive/2012/07/30/2591344.html
http://www.cnblogs.com/refactor/archive/2012/07/31/2591544.html
更新子文档:
在更新子文档时和查询时又有所不同(遇到数组时多加一个$符号):
testProvider.update({"comments.score":4},{'$set':{'comments.$.score':1}},{multi:true,w: 1},function(err, numberUpdated) {
console.log("numberUpdated: ", numberUpdated);
});
查询下可以发现结果已经改变了:
mongodb <wbr>子文档查询
如果我想往一个子文档的数组中插入一个子文档时可以使用push:
testProvider.update({"comments.score":1},{'$push':{'comments.$.others':{"name":"j","age":"24"}}},{multi:true,w: 1},function(err, numberUpdated) {
console.log("numberUpdated: ", numberUpdated);
});
查询之后可以发现comments.others里多了一个子文档,表明已经插入进去了:
mongodb <wbr>子文档查询
如果我想把刚刚插入的子文档删除时,可以使用pull:
testProvider.update({"comments.score":1},{'$pull':{'comments.$.others':{"name":"j"}}},{multi:true,w: 1},function(err, numberUpdated) {
console.log("numberUpdated: ", numberUpdated);
});
查询之后可以发现结果已经改变:
mongodb <wbr>子文档查询
注意:如下图所示pull值中的key对应的值必须为json格式的:
mongodb <wbr>子文档查询
mongodb <wbr>子文档查询
我在使用的过程中还发现在遇到数组时使用'.$'符号时每个里面只能用一次;如上图的错误写法中就含有2个'.$'符号。所以如果子文档里嵌套有2个乃至更多数组时,很难准确的对嵌套的数组中的某一个元素进行更新。
标签: it 分类: mongoDB
首先我的数据库里有两个文档,每个文档里都有2个嵌套的数组:mongodb <wbr>子文档查询
如果我想查询comments里score大于5的记录:
testProvider.find({"comments.score":{"$gt":5}},{},function(err, result){
console.log(result);
});
返回了查找到的记录,说明了查找有效果了:
mongodb <wbr>子文档查询
这样查找返回的数据都是把匹配的整个文档返回过来了。
如果我觉得其中有个字段是没必要返回给我的可以设置这个字段为0:
testProvider.find({"comments.score":{"$gt":5}},{'comments.others':0},function(err, result) {
console.log(JSON.stringify(result));
});
可以看到返回的结果中已经没有others这个字段了
mongodb <wbr>子文档查询
如果我想查询comments数组里面的某个子文档中others数组里面name为a的记录:
testProvider.find({"comments.others.name":'a'},{},function(err, result) {
console.log(JSON.stringify(result));
});
返回的结果中的一部分如下图:
mongodb <wbr>子文档查询
从上面的几个示例中可以看到我想要查找某个符合条件的子文档时可以直接一直用'.'号表示下去而不用管其中是否有数组,如:{"comments.others.name":'a'}
一些高级的查询语法看这一篇:http://blog.sina.com.cn/s/blog_abba9c1d0101eti7.html
1、使用"$size"可以查询指定长度的数组
查询数组长度为3的数组db.users.find({"emails":{"$size":3}})
常见的查询是数组长度范围的查询."$size"并不能与其他查询子句组合(如:"$gt"),但是这种查询可以通过
在文档中添加一个"size"键的方式来实现.这样每一次向指定数组添加元素的时候,同时增加"size"值.原来这样
的更新: db.users.update({"$push":{"emails":"295240648@139.com"}})
变成这样的更新: db.users.update({"$push":{"emails":"295240648@139.com"},"$inc":{"size":1}})
这样就可以这样查询了 db.users.find({"size":{"$gt":3}})
2、使用"$slice"查询
find的第二个参数是可选的,可以指定返回那些键,"$slice"返回数组的一个子集合
返回emails数组的前两个元素 db.users.find({"userName":"refactor"},{"emails":{"$slice":2}})
返回emails数组的后两个元素 db.users.find({"userName":"refactor"},{"emails":{"$slice":-2}})
返回emails数组的第2个和第11个元素.如果数组不够11个,则返回第2个后面的所有元素
db.users.find({"userName":"refactor"},{"emails":{"$slice":[1,10]}})
"$slice"默认将返回文档中的所有键.
3、要正确指定一组条件,而不用指定每个键,要使用"$elemMatch".这种模糊的命名条件能用来部分指定匹配数组中
的单个内嵌文档的限定条件.正确写法应该是:
db.blog.find(
{
"comments":
{
"$elemMatch":
{
"author":"refactor",
"score":{"$gte":5}
}
}
}
)
"$elemMatch"将限定条件进行分组,仅当需要对一个内嵌文档的多个键操作时才会用到.
4."$where"查询
"$where"可以执行任意javascript作为查询的一部分.这使得查询能做(几乎)任何事情.
最典型的应用就是比较文档中的两个键的值是否相等.
如:db.blog.insert({"title":"refactor","content":"refactor"})
db.blog.insert({"title":"refactor1","content":"refactor content"})
第一个文档title键和content键的值相同.要返回该文档.
db.blog.find(
{
"$where":function()
{
for(var current in this)
{
for(var other in this)
{
if(current!=other&&this[current]==this[other])
{
return true;
}
}
}
return false;
}
}
)
如果函数返回true,文档就作为结果的一部分被返回.
上面是用一个函数,也可以用一个字符串指定"$where"查询.下面两种方式是等价的:
db.blog.find({"$where":"this.x+this.y==10"})
db.blog.find({"$where":"function(){return this.x+this.y==10;}"})
不是非常必要时,应避免"$where"查询.它在速度上要比常规查询慢的多.每个文档都要从BSON转换成
javascript对象,然后通过"$where"的表达式来运行.它还不能利用索引.
将常规查询作为前置过滤,与"$where"组合使用才能不牺牲性能,如果有可能的话,用索引根据非"$where"
子句进行过滤,"$where"只用于对结果进行调优
上面的4个查询转自:http://www.cnblogs.com/refactor/archive/2012/07/30/2591344.html
http://www.cnblogs.com/refactor/archive/2012/07/31/2591544.html
更新子文档:
在更新子文档时和查询时又有所不同(遇到数组时多加一个$符号):
testProvider.update({"comments.score":4},{'$set':{'comments.$.score':1}},{multi:true,w: 1},function(err, numberUpdated) {
console.log("numberUpdated: ", numberUpdated);
});
查询下可以发现结果已经改变了:
mongodb <wbr>子文档查询
如果我想往一个子文档的数组中插入一个子文档时可以使用push:
testProvider.update({"comments.score":1},{'$push':{'comments.$.others':{"name":"j","age":"24"}}},{multi:true,w: 1},function(err, numberUpdated) {
console.log("numberUpdated: ", numberUpdated);
});
查询之后可以发现comments.others里多了一个子文档,表明已经插入进去了:
mongodb <wbr>子文档查询
如果我想把刚刚插入的子文档删除时,可以使用pull:
testProvider.update({"comments.score":1},{'$pull':{'comments.$.others':{"name":"j"}}},{multi:true,w: 1},function(err, numberUpdated) {
console.log("numberUpdated: ", numberUpdated);
});
查询之后可以发现结果已经改变:
mongodb <wbr>子文档查询
注意:如下图所示pull值中的key对应的值必须为json格式的:
mongodb <wbr>子文档查询
mongodb <wbr>子文档查询
我在使用的过程中还发现在遇到数组时使用'.$'符号时每个里面只能用一次;如上图的错误写法中就含有2个'.$'符号。所以如果子文档里嵌套有2个乃至更多数组时,很难准确的对嵌套的数组中的某一个元素进行更新。
阅读全文
0 0
- mongodb 子文档查询
- mongodb 子文档查询
- 查询MongoDB子文档的List字段
- Mongodb查询文档
- mongodb文档查询地址
- MongoDB 查询文档
- MongoDB 查询文档
- Mongodb 查询文档
- MongoDB 查询文档
- mongoDB 文档查询
- MongoDB 查询文档(四)
- MongoDB 查询文档(五)
- MongoDB 查询文档
- mongoDB子文档列表更新
- (MongoDB系列之合并MongoDB子文档
- 用MongoDB做子查询
- mongodb查询内嵌文档
- mongodb查询内嵌文档
- 【Objective-C】Base64
- SSM+Bootstrap场地管理系统第一天----数据库的建立和mybatis的逆向工程
- strcpy 函数
- jdk的安装与环静变量配置
- HBase安装手册
- mongodb 子文档查询
- ACM dfs回溯法 Prime Ring Problem
- 空域高斯滤波与频域高斯滤波
- iOS MDM相关资料
- Android 常用开源框架汇总
- Google Inception Net 特点总结
- EJSP技术理解
- Redis 使用笔记之 hash
- 字符串和JSON对象的互相转化