聚合操作能夠處理數(shù)據(jù)記錄并返回計(jì)算結(jié)果。聚合操作能將多個(gè)文檔中的值組合起來,對(duì)成組數(shù)據(jù)執(zhí)行各種操作,返回單一的結(jié)果。它相當(dāng)于 SQL 中的 count(*) 組合 group by。
對(duì)于 MongoDB 中的聚合操作,應(yīng)該使用 aggregate()
方法。
aggregate()
方法中的基本格式如下所示:
>db.COLLECTION_NAME.aggregate(AGGREGATE_OPERATION)
假如某個(gè)集合包含下列數(shù)據(jù):
{
_id: ObjectId(7df78ad8902c)
title: 'MongoDB Overview',
description: 'MongoDB is no sql database',
by_user: 'tutorials point',
url: 'http://www.tutorialspoint.com',
tags: ['mongodb', 'database', 'NoSQL'],
likes: 100
},
{
_id: ObjectId(7df78ad8902d)
title: 'NoSQL Overview',
description: 'No sql database is very fast',
by_user: 'tutorials point',
url: 'http://www.tutorialspoint.com',
tags: ['mongodb', 'database', 'NoSQL'],
likes: 10
},
{
_id: ObjectId(7df78ad8902e)
title: 'Neo4j Overview',
description: 'Neo4j is no sql database',
by_user: 'Neo4j',
url: 'http://www.neo4j.com',
tags: ['neo4j', 'database', 'NoSQL'],
likes: 750
},
假如想從上述集合中,歸納出一個(gè)列表,以顯示每個(gè)用戶寫的教程數(shù)量,需要像下面這樣使用 aggregate()
方法:
> db.mycol.aggregate([{$group : {_id : "$by_user", num_tutorial : {$sum : 1}}}])
{
"result" : [
{
"_id" : "tutorials point",
"num_tutorial" : 2
},
{
"_id" : "Neo4j",
"num_tutorial" : 1
}
],
"ok" : 1
}
>
假如用 SQL 來處理上述查詢,則需要使用這樣的命令:select by_user, count(*) from mycol group by by_user
。
上例使用 by_user 字段來組合文檔,每遇到一次 by_user
,就遞增之前的合計(jì)值。下面是聚合表達(dá)式列表。
表達(dá)式 | 描述 | 范例 |
---|---|---|
$sum |
對(duì)集合中所有文檔的定義值進(jìn)行加和操作 | db.mycol.aggregate([{$group : {_id : "$by_user", num_tutorial : {$sum : "$likes"}}}]) |
$avg |
對(duì)集合中所有文檔的定義值進(jìn)行平均值 | db.mycol.aggregate([{$group : {_id : "$by_user", num_tutorial : {$avg : "$likes"}}}]) |
$min |
計(jì)算集合中所有文檔的對(duì)應(yīng)值中的最小值 | db.mycol.aggregate([{$group : {_id : "$by_user", num_tutorial : {$min : "$likes"}}}]) |
$max |
計(jì)算集合中所有文檔的對(duì)應(yīng)值中的最大值 | db.mycol.aggregate([{$group : {_id : "$by_user", num_tutorial : {$max : "$likes"}}}]) |
$push |
將值插入到一個(gè)結(jié)果文檔的數(shù)組中 | db.mycol.aggregate([{$group : {_id : "$by_user", url : {$push: "$url"}}}]) |
$addToSet |
將值插入到一個(gè)結(jié)果文檔的數(shù)組中,但不進(jìn)行復(fù)制 | db.mycol.aggregate([{$group : {_id : "$by_user", url : {$addToSet : "$url"}}}]) |
$first |
根據(jù)成組方式,從源文檔中獲取第一個(gè)文檔。但只有對(duì)之前應(yīng)用過 $sort 管道操作符的結(jié)果才有意義。 |
db.mycol.aggregate([{$group : {_id : "$by_user", first_url : {$first : "$url"}}}]) |
$last |
根據(jù)成組方式,從源文檔中獲取最后一個(gè)文檔。但只有對(duì)之前進(jìn)行過 $sort 管道操作符的結(jié)果才有意義。 |
db.mycol.aggregate([{$group : {_id : "$by_user", last_url : {$last : "$url"}}}]) |
在 UNIX 命令 Shell 中,管道(pipeline)概念指的是能夠在一些輸入上執(zhí)行一個(gè)操作,然后將輸出結(jié)果用作下一個(gè)命令的輸入。MongoDB 的聚合架構(gòu)也支持這種概念。管道中有很多階段(stage),在每一階段中,管道操作符都會(huì)將一組文檔作為輸入,產(chǎn)生一個(gè)結(jié)果文檔(或者管道終點(diǎn)所得到的最終 JSON 格式的文檔),然后再將其用在下一階段。
聚合架構(gòu)中可能采取的管道操作符有: