使用聚合框架无法进行您描述的计算类型 - 这不是因为没有$unwind
非数组的方法。即使 person:value 对象是数组中的文档,$unwind
也无济于事。
“分组依据”功能(无论是在 MongoDB 中还是在任何关系数据库中)都是在字段或列的值上完成的。我们根据另一个字段的值按字段值和总和/平均值/等进行分组。
简单示例是您建议的变体,将评级字段添加到示例文章集合中,但不是作为从用户到评级的映射,而是作为这样的数组:
{ title : title of article", ...
ratings: [
{ voter: "user1", score: 5 },
{ voter: "user2", score: 8 },
{ voter: "user3", score: 7 }
]
}
现在您可以将其汇总为:
[ {$unwind: "$ratings"},
{$group : {_id : "$ratings.voter", averageScore: {$avg:"$ratings.score"} } }
]
但是这个例子的结构如你所描述的那样看起来像这样:
{ title : title of article", ...
ratings: {
user1: 5,
user2: 8,
user3: 7
}
}
甚至这个:
{ title : title of article", ...
ratings: [
{ user1: 5 },
{ user2: 8 },
{ user3: 7 }
]
}
即使你能做到$unwind
这一点,这里也没有什么可以汇总的。除非您知道所有可能的密钥(用户)的完整列表,否则您无法对此做太多事情。[*]
与您所拥有的类似的关系数据库模式是:
CREATE TABLE T (
user1: integer,
user2: integer,
user3: integer
...
);
这不是要做的事情,而是我们会这样做:
CREATE TABLE T (
username: varchar(32),
score: integer
);
现在我们使用 SQL 进行聚合:
select username, avg(score) from T group by username;
有一个对 MongoDB 的增强请求,可能允许您将来在聚合框架中执行此操作 - 将值投影到键的能力,反之亦然。同时,总是有map/reduce。
[*] 如果您知道所有唯一键(您可以使用与此类似的方法找到所有唯一键),则有一种复杂的方法可以做到这一点,但是如果您知道所有键,您也可以只运行一系列查询每个 userX 的表单db.articles.find({"ratings.user1":{$exists:true}},{_id:0,"ratings.user1":1})
将返回他们的所有评分,您可以简单地对它们进行求和和平均,而不是进行聚合框架所需的非常复杂的投影。