如何使用MongoDB聚合进行分页?

Joh*_*976 8 pagination mongodb aggregation-framework

我想执行一个执行基本分页的聚合查询:

  1. 查找属于某个特定的所有订单 company_id
  2. 按顺序排序 order_number
  3. 计算文件总数
  4. 跳到例如文档编号100并传递其余部分
  5. 将文档数量限制为例如2并传递它们
  6. 通过从文档中返回计数和选定的几个字段来完成

以下是查询的细分:

db.Order.collection.aggregate([
Run Code Online (Sandbox Code Playgroud)

这会找到所有匹配的文档:

  { '$match'    : { "company_id" : ObjectId("54c0...") } },
Run Code Online (Sandbox Code Playgroud)

这会对文件进行排序:

  { '$sort'     : { 'order_number' : -1 } },
Run Code Online (Sandbox Code Playgroud)

这会对文档进行计数并传递未经修改的文档,但我确信这样做是错误的,因为事情变得怪异了:

  {
    '$group' : {
      '_id'     : null,
      'count'   : { '$sum' : 1 },
      'entries' : { '$push' : "$$ROOT" }
    }
  },
Run Code Online (Sandbox Code Playgroud)

这似乎跳过一些文件:

  { "$skip"     : 100 },
Run Code Online (Sandbox Code Playgroud)

这应该限制文档,但它不会:

  { "$limit"    : 2 },
Run Code Online (Sandbox Code Playgroud)

这会返回计数,但它不会返回数组中的文档,而是返回包含每个字段的数组:

  { '$project'  : {
      'count'     : 1,
      'entries'   : {'_id' : "$entries._id", 'order_number' : "$entries.order_number"}
    }
  }
])
Run Code Online (Sandbox Code Playgroud)

这是结果:

[
  { "_id" : null,
    "count" : 300,
    "entries" : [
      {
        "_id" : [ObjectId('5a5c...'), ObjectId('5a5c...')],
        "order_number" : ["4346", "4345"]
      },
      {
        "_id" : [ObjectId('5a5c...'), ObjectId('5a5c...')],
        "order_number" : ["4346", "4345"]
      },
      ...
    ]
  }
]
Run Code Online (Sandbox Code Playgroud)

我在哪里弄错了?

Ale*_*lex 22

要计算总计返回子集,您需要对同一数据集应用分组和跳过/限制.为此你可以利用方面

例如,要显示第3页,每页10个文档:

db.Order.aggregate([
    { '$match'    : { "company_id" : ObjectId("54c0...") } },
    { '$sort'     : { 'order_number' : -1 } },
    { '$facet'    : {
        metadata: [ { $count: "total" }, { $addFields: { page: NumberInt(3) } } ],
        data: [ { $skip: 20 }, { $limit: 10 } ] // add projection here wish you re-shape the docs
    } }
] )
Run Code Online (Sandbox Code Playgroud)

它将返回包含2个字段的单个文档:

{
    "metadata" : [ 
        {
            "total" : 300,
            "page" : 3
        }
    ],
    "data" : [ 
        {
            ... original document ...
        }, 
        {
            ... another document ...
        }, 
        {
            ... etc up to 10 docs ...
        }
    ]
}
Run Code Online (Sandbox Code Playgroud)

  • @ JohnSmith1976只是一个例子。假设每页10个文档。第一页返回前10个文件。为此,您将跳过0。对于第n页,您跳过`(n-1)* pageSize`,因此对于第3页,则跳过`(3-1)* 10 = 20`。 (2认同)

nim*_*rok 10

自 mongoDB 5.0 版本以来,还有另一个选项,可以避免将$facet所有返回的文档分组为一个大文档的缺点。主要关心的是一个文件的大小限制为16M。使用$setWindowFields可以避免这种问题:

db.Order.aggregate([
    {$match: {company_id: ObjectId("54c0...") } },
    {$sort: {order_number: -1 } },
    {$setWindowFields: {output: {totalCount: {$count: {}}}}}
    {$skip: 20 },
    {$limit: 10 } 
])
Run Code Online (Sandbox Code Playgroud)