多个mongoose count()查询到MongoDB

时间:2013-12-18 08:46:16

标签: javascript node.js mongodb mongoose

首先:我 - > MongoNoob和我知道这已经以一种或另一种方式被问到了,但直到现在我还没有找到任何具体内容。 假设我有两个这样描述的Moongoose模型:

var pollSchema = mongoose.Schema({
    title: String,
    choices: [{
        content: String
    }]
});
var choiceSchema = mongoose.Schema({
    poll_id: mongoose.Schema.ObjectId,
    option: Number
});

UI显示轮询,当用户选择一个选项时,会将其写入choiceSchema模型。现在我想创建一个'统计',告诉我有多少用户选择了选项1,选项2,选项3,.... 我可以简单地获取find轮询的所有选项,并在服务器代码中生成统计信息,但如果我有一百万用户选择,我将不得不处理相同大小的数组。这不可能是对的。 然而,我可以生成查询并使用count()方法:

var query = Choice.find({poll_id: someId}),
query.where('option', 1);
var resultForOption1;
query.count(function(err, count)) {
    resultForOption1 = count;
});

如何为多个选项执行此操作并将结果“加入”(哈哈)到数组中?由于这是异步的,我会嵌套调用,但这不是可变数量的查询的选项。

我是否想念树木:-)?有人能指出我正确的方向吗?

BR, 丹尼尔

2 个答案:

答案 0 :(得分:4)

您可以使用aggregate执行此操作:

Choice.aggregate(
    {$group: {_id: {poll_id: '$poll_id', option: '$option'}, count: {$sum: 1}}}
).exec(...);

这将按choicespoll_idoption集合文档进行分组,并计算每个文档的出现次数,并提供如下输出:

{
  "result": [
    {
      "_id": {
        "poll_id": 2,
        "option": 3
      },
      "count": 1
    },
    {
      "_id": {
        "poll_id": 1,
        "option": 2
      },
      "count": 2
    },
    {
      "_id": {
        "poll_id": 2,
        "option": 2
      },
      "count": 1
    },
    {
      "_id": {
        "poll_id": 1,
        "option": 1
      },
      "count": 1
    }
  ],
  "ok": 1
}

然后,您可以使用$group管道中的后续$projectaggregate阶段,根据需要进一步对生成的文档进行分组和重新整形。

答案 1 :(得分:2)

如果您只想迭代count查询并轻松清理,您可以使用像co这样的库,甚至更好,async:

async.map({ "option":1 },{ "option":2 }], Choice.find, function(e, r){
    // And you continue from here
});

另一种选择 - 在我看来更好 - 。是使用aggregate queries。您将了解有关mongoDB的更多信息,并且应该更有效率。在我发给你的链接中,第一个例子几乎就是你想做的事情:

query