注:本文基于MongoDB 4.2.6编写
> db.bb.find()
{ "_id" : ObjectId("5ec29c1e6b02bc57526eb598"), "hello" : "ball", "animals" : { "pig" : 80, "cow" : 200, "dog" : 50 }, "may" : "May", "happy" : "ending" }
{ "_id" : ObjectId("5ec5384c6d59a15615f7df04"), "hello" : "ball", "name" : "jan", "happy" : "ending" }
为了便于阅读,我们可以对结果进行优化显示,
> db.bb.find().pretty()
{
"_id" : ObjectId("5ec29c1e6b02bc57526eb598"),
"hello" : "ball",
"animals" : {
"pig" : 80,
"cow" : 200,
"dog" : 50
},
"may" : "May",
"happy" : "ending"
}
{
"_id" : ObjectId("5ec5384c6d59a15615f7df04"),
"hello" : "ball",
"name" : "jan",
"happy" : "ending"
}
> db.bb.find({"name": "jan"}).pretty()
{
"_id" : ObjectId("5ec5384c6d59a15615f7df04"),
"hello" : "ball",
"name" : "jan",
"happy" : "ending"
}
> db.bb.find({"name": "jan"}, {"hello": 1, "name": 1}).pretty()
{
"_id" : ObjectId("5ec5384c6d59a15615f7df04"),
"hello" : "ball",
"name" : "jan"
}
有一点需要注意的是,不能同时制定哪些返回和哪些不返回,否则会报错,
> db.bb.find({"name": "jan"}, {"hello": 0, "_id": 1}).pretty()
Error: error: {
"ok" : 0,
"errmsg" : "Projection cannot have a mix of inclusion and exclusion.",
"code" : 2,
"codeName" : "BadValue"
}
> db.bb.find().pretty()
{
"_id" : ObjectId("5ec684c3b97b4da914bd5e67"),
"name" : "mike",
"score" : 90
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e68"),
"name" : "john",
"score" : 90
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e69"),
"name" : "lucy",
"score" : 100
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96
}
> db.bb.find({score: {$gt: 90, $lt: 100}}).pretty()
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96
}
> db.bb.find({score: {$in:[96, 100]}}).pretty()
{
"_id" : ObjectId("5ec684fab97b4da914bd5e69"),
"name" : "lucy",
"score" : 100
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96
}
> db.bb.find({$or: [{name: "jack"}, {score: 96}]}).pretty()
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96
}
> db.bb.find({name: /ja.*/}).pretty()
{
"_id" : ObjectId("5ec5384c6d59a15615f7df04"),
"hello" : "ball",
"name" : "jan",
"happy" : "ending"
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96
}
> db.bb.find({sport: "run"}).pretty()
{
"_id" : ObjectId("5ec697c0b97b4da914bd5e6b"),
"sport" : [
"run",
"swin",
"jump",
"tennis"
]
}
> db.bb.find({sport: {$all: ["run", "swin"]}}).pretty()
{
"_id" : ObjectId("5ec697c0b97b4da914bd5e6b"),
"sport" : [
"run",
"swin",
"jump",
"tennis"
]
}
但是如果数组中有多个元素,如下,记录了三人两次考试成绩,
> db.bb.find().pretty()
{
"_id" : ObjectId("5ed6554230571733ccb3d67b"),
"players" : [
{
"name" : "aa",
"score" : 98
},
{
"name" : "dan",
"score" : 95
},
{
"name" : "du",
"score" : 100
}
]
}
{
"_id" : ObjectId("5ed6579b30571733ccb3d67c"),
"players" : [
{
"name" : "aa",
"score" : 99
},
{
"name" : "dan",
"score" : 98
},
{
"name" : "du",
"score" : 90
}
]
}
如果需要找出du同学考试成绩大于95分的记录,
> db.bb.find({players: {name: "du", score: {$gt: 95}}}).pretty()
>
这种方式是无法查找出记录的,因为对于内嵌文档需要整个文档匹配,因此这是不可行的,
> db.bb.find({"players.name": "du", "players.score": {$gt: 95}}).pretty()
{
"_id" : ObjectId("5ed6554230571733ccb3d67b"),
"players" : [
{
"name" : "aa",
"score" : 98
},
{
"name" : "dan",
"score" : 95
},
{
"name" : "du",
"score" : 100
}
]
}
{
"_id" : ObjectId("5ed6579b30571733ccb3d67c"),
"players" : [
{
"name" : "aa",
"score" : 99
},
{
"name" : "dan",
"score" : 98
},
{
"name" : "du",
"score" : 90
}
]
}
>
而直接使用内嵌文档字段查询,也是有问题的,因为这相当于在整个内嵌文档中进行匹配,由于两个文档中都有du同学,同时都有同学的分数大于95,因此返回了两个文档。
这时候就要需要$elemMatch,它可以匹配内嵌文档中单个文档,也就是能将查询条件分组,以组作为查询条件就不会出现刚才的情况,
> db.bb.find({players: {$elemMatch: {name: "du", score: {$gt: 95}}}}).pretty()
{
"_id" : ObjectId("5ed6554230571733ccb3d67b"),
"players" : [
{
"name" : "aa",
"score" : 98
},
{
"name" : "dan",
"score" : 95
},
{
"name" : "du",
"score" : 100
}
]
}
> db.bb.find().pretty()
{
"_id" : ObjectId("5ec697c0b97b4da914bd5e6b"),
"sport" : [
"run",
"swin",
"jump",
"tennis"
],
"name" : "geek"
}
> db.bb.find({name: "geek"}, {sport: {$slice: -1}}).pretty()
{
"_id" : ObjectId("5ec697c0b97b4da914bd5e6b"),
"sport" : [
"tennis"
],
"name" : "geek"
}
> db.bb.find({name: "geek"}, {sport: {$slice: [2,3]}}).pretty()
{
"_id" : ObjectId("5ec697c0b97b4da914bd5e6b"),
"sport" : [
"jump",
"tennis"
],
"name" : "geek"
}
> db.bb.find({name: /^j.*/}).pretty()
{
"_id" : ObjectId("5ec5384c6d59a15615f7df04"),
"hello" : "ball",
"name" : "jan",
"happy" : "ending",
"email" : "com"
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e68"),
"name" : "john",
"score" : 90
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96,
"email" : "com"
}
> db.bb.find({name: /^j.*/}).limit(2).pretty()
{
"_id" : ObjectId("5ec5384c6d59a15615f7df04"),
"hello" : "ball",
"name" : "jan",
"happy" : "ending",
"email" : "com"
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e68"),
"name" : "john",
"score" : 90
}
> db.bb.find({name: /^j.*/}).skip(2).pretty()
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96,
"email" : "com"
}
> db.bb.find({name: /^j.*/}).sort({name: 1}).pretty()
{
"_id" : ObjectId("5ec684fab97b4da914bd5e6a"),
"name" : "jack",
"score" : 96,
"email" : "com"
}
{
"_id" : ObjectId("5ec5384c6d59a15615f7df04"),
"hello" : "ball",
"name" : "jan",
"happy" : "ending",
"email" : "com"
}
{
"_id" : ObjectId("5ec684fab97b4da914bd5e68"),
"name" : "john",
"score" : 90
}
其中,sort参数中指定对某个字段进行升序(1)或者降序(-1)排序,
基本上find上能使用的查找语法在update更新操作上一样能使用。