Ubi*_*ers 5 mongodb mongodb-query aggregation-framework
我试图根据一些匹配条件获取数据.首先我试过这个:这里ending_date是全日期格式
Offer.aggregate([
{
$match: {
carer_id : req.params.carer_id,
status : 3
}
},
{
$group : {
_id : { year: { $year : "$ending_date" }, month: { $month : "$ending_date" }},
count : { $sum : 1 }
}
}],
function (err, res)
{ if (err) ; // TODO handle error
console.log(res);
});
Run Code Online (Sandbox Code Playgroud)
这给了我以下输出:
[ { _id: { year: 2015, month: 11 }, count: 2 } ]
Run Code Online (Sandbox Code Playgroud)
现在我想检查一年,所以我正在尝试这个:
Offer.aggregate([
{
$project: {
myyear: {$year: "$ending_date"}
}
},
{
$match: {
carer_id : req.params.carer_id,
status : 3,
$myyear : "2015"
}
},
{
$group : {
_id : { year: { $year : "$ending_date" }, month: { $month : "$ending_date" }},
count : { $sum : 1 }
}
}],
function (err, res)
{ if (err) ; // TODO handle error
console.log(res);
});
Run Code Online (Sandbox Code Playgroud)
这给了我以下输出:
[]
Run Code Online (Sandbox Code Playgroud)
正如你所看到的,_id有2015年作为一年,所以当我匹配年份它应该是阵列.但我得到空数组.为什么这个?
有没有其他方法只匹配年份整个日期时间?
这是样本数据
{
"_id": {
"$oid": "56348e7938b1ab3c382d3363"
},
"carer_id": "55e6f647f081105c299bb45d",
"user_id": "55f000a2878075c416ff9879",
"starting_date": {
"$date": "2015-10-15T05:41:00.000Z"
},
"ending_date": {
"$date": "2015-11-19T10:03:00.000Z"
},
"amount": "850",
"total_days": "25",
"status": 3,
"is_confirm": false,
"__v": 0
}
{
"_id": {
"$oid": "563b5747d6e0a50300a1059a"
},
"carer_id": "55e6f647f081105c299bb45d",
"user_id": "55f000a2878075c416ff9879",
"starting_date": {
"$date": "2015-11-06T04:40:00.000Z"
},
"ending_date": {
"$date": "2015-11-16T04:40:00.000Z"
},
"amount": "25",
"total_days": "10",
"status": 3,
"is_confirm": false,
"__v": 0
}
Run Code Online (Sandbox Code Playgroud)
Bla*_*ven 13
你在这里做了很多错事,它确实需要一个解释,所以希望你能学到一些东西.
这是最基本的概念,但人们最常接受(甚至在继续使用之后)聚合"管道"就是这样,"管道"流程将输入馈送到每个阶段沿.想想"unix管道" |:
ps -ef | grep mongo | tee out.txt
Run Code Online (Sandbox Code Playgroud)
你已经看过类似的东西,毫无疑问,这是基本的概念,从第一件事输出到下一件事然后操纵给下一件事等的输入.
所以你要问的是基本问题:
{
$project: {
myyear: {$year: "$ending_date"}
}
},
{
$match: {
carer_id : req.params.carer_id,
status : 3,
$myyear : "2015"
}
},
Run Code Online (Sandbox Code Playgroud)
考虑一下$project这里有什么.您可以在输出中指定所需的字段,并将其"吐出",并可能通过操作.它是否在"添加"文档中的字段中输出这些字段?不,它没有.只有你要求出来的东西实际上出来了,可以在以下的管道阶段使用.
在$match这里主要询问的是不再存在领域,因为你只要求输出的一件事.同样的问题进一步发生,因为你再次询问你之前删除的字段,并且没有什么可以引用,但是一切都已经被一个$match无法匹配的东西删除了.
此外,这不是您输入的字段投影的工作方式
{ "$match": {
"carer_id" : req.params.carer_id,
"status" : 3,
"ending_date": {
"$gte": new Date("2015-01-01"),
"$lt": new Date("2016-01-01")
}
}},
{ "$group": {
"_id": {
"year": { "$year": "$ending_date" },
"month": { "$month": "$ending_date" }
},
"count": { "$sum": 1 }
}}
Run Code Online (Sandbox Code Playgroud)
为什么?因为它才有意义.如果您想匹配"年份",则提供全年的日期范围.我们可以$redact在提取的年份值上进行匹配,但这只是浪费处理时间.
这样做是最快的,并且实际上可以使用索引来更快地处理.所以不要解决问题,只要求你想要的日期范围.
您忘记投影您正在使用的字段$match以及$group稍后使用的字段.要快速修复,请改用此查询:
Offer.aggregate([
{
$project: {
myyear: { $year: "$ending_date" },
carer_id: 1,
status: 1,
ending_date: 1
}
},
{
$match: {
carer_id: req.params.carer_id,
myyear: 2015,
status: 3
}
},
{
$group: {
_id: {
year: { $year: "$ending_date" },
month: { $month: "$ending_date" }
},
count: { $sum: 1 }
}
}],
function (err, res)
{
if (err) {} // TODO handle error
console.log(res);
});
Run Code Online (Sandbox Code Playgroud)
也就是说,Blakes Seven解释了如何在她的回答中提出更好的质询.我认为你应该尝试使用她的方法.
| 归档时间: |
|
| 查看次数: |
18170 次 |
| 最近记录: |