MongoDB
 sql >> Base de données >  >> NoSQL >> MongoDB

Tableau de groupe agrégé MongoDB à la clé :valeur de somme

Pour votre formulaire réel, et donc en supposant que vous connaissiez réellement les valeurs possibles pour "type", vous pouvez le faire avec deux $group étapes et une certaine utilisation du $cond opérateur :

db.types.aggregate([
    { "$group": {
         "_id": {
             "stat": "$stat",
             "type": "$type"
         },
         "count": { "$sum": 1 }
    }},
    { "$group": {
        "_id": "$_id.stat",
        "type1": { "$sum": { "$cond": [
            { "$eq": [ "$_id.type", 1 ] },
            "$count",
            0
        ]}},
        "type2": { "$sum": { "$cond": [
            { "$eq": [ "$_id.type", 2 ] },
            "$count",
            0
        ]}},
        "type3": { "$sum": { "$cond": [
            { "$eq": [ "$_id.type", 3 ] },
            "$count",
            0
        ]}}
    }}
])

Ce qui donne exactement :

{ "_id" : "foobar", "type1" : 2, "type2" : 1, "type3" : 1 }

Je préfère en fait la forme plus dynamique avec deux $group étapes cependant :

db.types.aggregate([
    { "$group": {
         "_id": {
             "stat": "$stat",
             "type": "$type"
         },
         "count": { "$sum": 1 }
    }},
    { "$group": {
        "_id": "$_id.stat",
        "types": { "$push": {
            "type": "$_id.type",
            "count": "$count"
        }}
    }}
])

Pas la même sortie mais fonctionnelle et flexible aux valeurs :

{
    "_id" : "foobar",
    "types" : [
            {
                    "type" : 3,
                    "count" : 1
            },
            {
                    "type" : 2,
                    "count" : 1
            },
            {
                    "type" : 1,
                    "count" : 2
            }
    ]
}

Sinon, si vous avez besoin du même format de sortie mais que vous avez besoin des champs flexibles, vous pouvez toujours utiliser mapReduce, mais ce n'est pas exactement la même sortie.

db.types.mapReduce(
    function () {

        var obj = { };

        var key = "type" + this.type;
        obj[key] = 1;

        emit( this.stat, obj );

    },
    function (key,values) {

        var obj = {};

        values.forEach(function(value) {
            for ( var k in value ) {
                if ( !obj.hasOwnProperty(k) )
                    obj[k] = 0;
                obj[k]++;
            }
        });

        return obj;

    },
    { "out": { "inline": 1 } }
)

Et dans le style typique de mapReduce :

   "results" : [
            {
                    "_id" : "foobar",
                    "value" : {
                            "type1" : 2,
                            "type2" : 1,
                            "type3" : 1
                    }
            }
    ],

Mais ce sont vos options