Mongodb geoNear和组聚合

时间:2015-06-29 12:03:11

标签: javascript mongodb mongoose aggregation-framework

我一直在网上搜索与此相关的内容,但不能。

我有这种聚合

 Place.aggregate(
    [
        { "$geoNear": {
            "near": {
                "type": "Point",
                "coordinates": [longitude, latitude]
            },
            "spherical": true,
            "distanceField": "distance"
        }},
        { $group:
        {   _id: "$_id",
            name: { '$first': '$name' },
            distance: { $first: "$distance" }
        }
        },
        { $project : {
            name: 1,
            distance: 1,
        }}
    ],
    function(error, places) {
        if (error) return callback(error, null);
        callback(null, places)
    }
);

它有效,但geoNear排序丢失了!

但这给了我正确排序的文件:

    Place.aggregate(
    [
        { "$geoNear": {
            "near": {
                "type": "Point",
                "coordinates": [longitude, latitude]
            },
            "spherical": true,
            "distanceField": "distance"
        }}
    ],
    function(error, places) {
        if (error) return callback(error, null);
        callback(null, places)
    }
);

有什么想法吗?

为了让您了解我在这里尝试做的是使用

的完整查询
    Place.aggregate(
    [
        { "$geoNear": {
            "near": {
                "type": "Point",
                "coordinates": [longitude, latitude]
            },
            "spherical": true,
            "distanceField": "distance"
        }},
        {"$unwind": "$participants" } ,
        { $group:
        {   _id: "$_id",
            name: { '$first': '$name' },
            distance: { $first: "$distance" },
            person: { $sum: 1 },
            sumof:{ $sum: "$participants.age"}
        }
        },
        { $project : {
            name: name,
            distance: 1,
            person: 1,
            meanAge:{ $divide: [ "$sumof", "$person" ]}
        }}
    ],
    function(error, places) {
        if (error) return callback(error, null);
        callback(null, places)
    }
);

1 个答案:

答案 0 :(得分:3)

简而言之,当您使用$group之类的运算符时,无法保证返回结果的顺序。文档将按照对组管道进行“输入”的顺序进行处理,以便遵守$first之类的内容,但输出的输出顺序不一定与输入的顺序相同。

直接来自文档:

  

$ group不会对其输出文档进行排序。

事实上,您可能会发现订单是通过分组键,但在大多数情况下都是相反的。

如果你想要一个特定的输出顺序,那么使用$sort和最后的“输出”应该是你的最后一个管道阶段,所以没有其他任何改变这个顺序。

Place.aggregate(
    [
        { "$geoNear": {
            "near": {
                "type": "Point",
                "coordinates": [longitude, latitude]
            },
            "spherical": true,
            "distanceField": "distance"
        }},
        { "$unwind": "$participants" } ,
        { "$group": {   
            "_id": "$_id",
             "name": { "$first": "$name" },
             "distance": { "$first": "$distance" },
             "person": { "$sum": 1 },
             "sumof":{ "$sum": "$participants.age" }
        }},
        { "$project" : {
            "name": 1,
            "distance": 1,
            "person": 1,
            "meanAge": { "$divide": [ "$sumof", "$person" ]}
        }},
        { "$sort": { "distance": 1 } }
    ],
    callback
 );