PHP& MongoDB显示按日期分组的结果

时间:2015-02-11 12:20:39

标签: php mongodb aggregation-framework

我有一个包含以下文件的mongodb集合:

{
  "_id" : ObjectId("547af6aea3f0eba7148b4567"),
  "check_id" : "f5d654e7-257d-4a93-ae50-2d59dfeeb451",
  "chunks" : NumberLong(200),
  "num_hosts" : NumberLong(1000),
  "num_rbls" : NumberLong(163),
  "owner" : NumberLong(7901),
  "created" : ISODate("2014-11-30T10:51:26.924Z"),
  "started" : ISODate("2014-11-30T10:51:31.558Z"),
  "finished" : ISODate("2014-11-30T10:57:08.512Z")
}

{
  "_id" : ObjectId("54db19a858a5d395a18b4567"),
  "check_id" : "9660e510-1349-43f3-9d5e-8bf4b06179be",
  "chunks" : NumberLong(2),
  "num_hosts" : NumberLong(10),
  "num_rbls" : NumberLong(166),
  "owner" : NumberLong(7901),
  "created" : ISODate("2015-02-11T08:58:17.118Z"),
  "started" : ISODate("2015-02-11T08:58:18.78Z"),
  "finished" : ISODate("2015-02-11T08:58:47.486Z")
}


{
  "_id" : ObjectId("54db267758a5d30eab8b4567"),
  "check_id" : "9660e510-1349-43f3-9d5e-8bf4b06179be",
  "chunks" : NumberLong(2),
  "num_hosts" : NumberLong(10),
  "num_rbls" : NumberLong(166),
  "owner" : NumberLong(7901),
  "created" : ISODate("2015-02-11T09:52:55.388Z"),
  "started" : ISODate("2015-02-11T09:52:56.109Z"),
  "finished" : ISODate("2015-02-11T09:53:22.095Z")
}

我需要的是获得结果并生成类似于此的数组:

Array
(
    [2015-02-11] => array
        (
            //array with results from 2015-02-11
        )
    [2014-11-30] => array
        (
            //array with results from 2014-11-30
        )
)

我知道可以只执行简单的收集 - >查找然后循环结果并使用php逻辑来实现我的目标但是有可能使用mongo吗?也许使用聚合框架?

编辑:我想按"创建"分组结果日期

任何帮助都将受到高度赞赏。

2 个答案:

答案 0 :(得分:2)

Monogo聚合mongo aggregation group用于此,因此以下查询可能会解决您的问题

db.collectionName.aggregate({
    "$group": {
    "_id": "$created",
    "data": {
        "$push": {
            "check_id": "$check_id",
            "chunks": "$chunks",
            "num_hosts": "$num_hosts",
            "num_rbls": "$num_rbls",
            "owner": "$owner",
            "started": "$started",
            "finished": "$finished"
        }
    }
    }
}).pretty()

db.collectionName.aggregate({
    "$group": {
    "_id": "$created",
    "data": {
        "$push": "$$ROOT"
    }
    }
}).pretty()

同样在mongo 2.8 $dateToString提供了将ISO日期转换为字符串格式的功能,所以下面的查询也可以正常工作

db.collectionName.aggregate([
    {
    "$project": {
        "yearMonthDay": {
            "$dateToString": {
                "format": "%Y-%m-%d",
                "date": "$created"
            }
        },
        "check_id": "$check_id",
        "chunks": "$chunks",
        "num_hosts": "$num_hosts",
        "num_rbls": "$num_rbls",
        "owner": "$owner",
        "started": "$started",
        "finished": "$finished"
    }
    },
    {
    "$group": {
        "_id": "$yearMonthDay",
        "data": {
            "$push": "$$ROOT"
        }
    }
    }
]).pretty()

答案 1 :(得分:0)

我已经设法使用聚合框架解决了这个问题。这是答案,万一有人需要它。

$op = array(
    array(
        '$project' => array(
            'data' => array(
                'check_id' => '$check_id',
                'chunks' => '$chunks',
                'num_hosts' => '$num_hosts',
                'num_rbls' => '$num_rbls',
                'owner' => '$owner',
                'started' => '$started',
                'finished' => '$finished',
            ),
            'year' => array('$year' => '$created' ),
            'month' => array('$month' => '$created' ),
            'day' => array('$dayOfMonth' => '$created'),
        )
    ),
    array(
        '$group' => array(
            '_id' => array('year' => '$year', 'month' => '$month', 'day' => '$day'),
            'reports_data' => array('$push' => '$data'),
        )
    ),
);

$c = $collection->aggregate($op);