51工具盒子

依楼听风雨
笑看云卷云舒,淡观潮起潮落

MongoDB聚合以多个字段分组。

英文:

MongoDB aggregation to group by multiple fields

问题 {#heading}

以下是您要翻译的内容:

我在我的集合中有以下数据

{ "name": "test", "data": { "statusOne": "enabled", "statusTwo": "active" } } { "name": "test", "data": { "statusOne": "disabled", "statusTwo": "active" } } { "name": "another-test", "data": { "statusOne": "disabled", "statusTwo": "active" } }

如何编写聚合查询以显示以下数据。需要按名称,statusOne和statusTwo进行分组。但首先基于名称字段对结果进行分隔。然后计算statusOne和statusTwo的出现次数。这两个结果都需要放入相同的输出字段"data"中。

"output": [
{
	"name": "test",
	"data": [
		{
			"status": "active",
			"count": 2
		},
		{
			"status": "disabled",
			"count": 1
		},
		{
			"status": "enabled",
			"count": 1
		}
	]
},
{
	"name": "another-test",
	"data": [
		{
			"status": "active",
			"count": 1
		},
		{
			"status": "disabled",
			"count": 1
		}
	]
}
]

尝试使用如下所述的分组序列,但没有成功

[
  {
    $group: {
      _id: {
        appName: "$name",
        cs: "$data.statusOne",
        ps: "$data.statusTwo,
      },
      total: {
        $sum: 1,
      },
    },
  },
  {
    $group: {
      _id: "$_id.name",
      total: { $sum: "$total" },
      ps: {
        $addToSet: {
          name: "$_id.ps",
          count: "$total",
        },
      },
      cs: {
        $addToSet: {
          name: "$_id.cs",
          count: "$total",
        },
      },
    },
  },
  {
    $project: {
      _id: 0,
      appName: "$_id",
      items: {
        $concatArrays: ["$ps", "$cs"],
      },
    },
  },
]

请注意,上述代码示例中的引号可能需要根据您的编程环境进行调整。 英文:

I have following data in my collection

{
  "name": "test",
  "data": {
    "statusOne": "enabled",
    "statusTwo": "active"
  }
}
{
  "name": "test",
  "data": {
    "statusOne": "disabled",
    "statusTwo": "active"
  }
}
{
  "name": "another-test",
  "data": {
    "statusOne": "disabled",
    "statusTwo": "active"
  }
}

How to write an aggregation query to display the data like below. Need to group by name, statusOne, statusTwo. But segregate the result first based on name field. Then calculate the number of occurrences of statusOne and statusTwo. Both the results need to go into the same output field "data"

"output": [
{
	"name": "test",
	"data": [
		{
			"status": "active",
			"count": 2
		},
		{
			"status": "disabled",
			"count": 1
		},
		{
			"status": "enabled",
			"count": 1
		}
	]
},
{
	"name": "another-test",
	"data": [
		{
			"status": "active",
			"count": 1
		},
		{
			"status": "disabled",
			"count": 1
		}
	]
}
]

Tried to use group by sequence as mentioned here but no luck

[
  {
    $group: {
      _id: {
        appName: "$name",
        cs: "$data.statusOne",
        ps: "$data.statusTwo,
      },
      total: {
        $sum: 1,
      },
    },
  },
  {
    $group: {
      _id: "$_id.name",
      total: { $sum: "$total" },
      ps: {
        $addToSet: {
          name: "$_id.ps",
          count: "$total",
        },
      },
      cs: {
        $addToSet: {
          name: "$_id.cs",
          count: "$total",
        },
      },
    },
  },
  {
    $project: {
      _id: 0,
      appName: "$_id",
      items: {
        $concatArrays: ["$ps", "$cs"],
      },
    },
  },
]

答案1 {#1}

得分: 2

db.collection('表名').aggregate([
  {
    $group: {
      _id: { name: "$name", status: "$data.statusOne" },
      count: { $sum: 1 }
    }
  },
  {
    $group: {
      _id: "$_id.name",
      data: {
        $push: {
          status: "$_id.status",
          count: "$count"
        }
      }
    }
  },
  {
    $project: {
      _id: 0,
      name: "$_id",
      data: 1
    }
  }
])

英文:

db.collection('tablename').aggregate([
  {
    $group: {
      _id: { name: "$name", status: "$data.statusOne" },
      count: { $sum: 1 }
    }
  },
  {
    $group: {
      _id: "$_id.name",
      data: {
        $push: {
          status: "$_id.status",
          count: "$count"
        }
      }
    }
  },
  {
    $project: {
      _id: 0,
      name: "$_id",
      data: 1
    }
  }
])

use this and change the collection name .

答案2 {#2}

得分: 1

这是一个通用的解决方案:

db.foo.aggregate([
    {$project: {
        name: true,
    /*                                                                       
    我们可以通过$objectToArray的输出进行更高级的操作,例如仅获取statusOne和statusTwo:                                 
   Z: {$filter: {                                                        
       input: {$objectToArray: "$data"},                                 
       cond: {$in: ['$$this.k', ['statusOne','statusTwo']]}              
       }}                                                                
                                                                          
   或者仅查找以'status'开头的键:                    
                                                                          
   Z: {$filter: {                                                        
       input: {$objectToArray: "$data"},                                 
       cond: {$eq: ['status', {$substr:['$$this.k',0,6]}]}               
       }}                                                                
                                                                          
  现在让我们简单点,获取整个内容而不进行过滤。                                                     
*/

Z: {$objectToArray: "$data"}

}} ,{$unwind: "$Z"}

// 现在我们有了这样的文档:
//{
// _id: ObjectId("64d651283d8bc34d3928366d"),
// name: 'another-test',
// Z: {
// k: 'statusOne',
// v: 'disabled'
// }
//}

// 现在只是分组和重新组织的问题:
,{$group: {_id: {"name":"$name", "v":"$Z.v"}, N:{$sum:1}}} ,{$group: {_id: "$_id.name", data: {$push: {status:"$_id.v", count:"$N"}} }}

]);

产生的结果为:

{
  _id: 'test',
  data: [
    {
      status: 'active',
      count: 2
    },
    {
      status: 'enabled',
      count: 1
    },
    {
      status: 'disabled',
      count: 1
    }
  ]
}
{
  _id: 'another-test',
  data: [
    {
      status: 'disabled',
      count: 1
    },
    {
      status: 'active',
      count: 1
    }
  ]
}

如果有成千上万的测试,我们不想使用$unwind怎么办? {#unwind}

在管道中使用$unwind时,必须小心。如果每个测试的平均状态数量变大(比如100),那么在管道中将会有大量文档。以下是一种利用$reduce的替代解决方案:

c=db.foo.aggregate([
    {$project: {
	name: true,
	Z: {$objectToArray: "$data"}
    }}
/*
 不使用$unwind并使用$group-$sum来计数,让我们自己来做。
 我们将使用新的Z覆盖旧的Z。
 此外,由于这是一个中间步骤,使用较短的变量名称以增加清晰度。

这个if-then-else构造基本上是这样的: 对于每种状态类型,status_type_count += 1 这允许输入具有相同类型的多个状态,例如:

"name": "test", "data": { "statusOne": "enabled", "statusTwo": "enabled" }

在MQL中,$reduce循环中,我们不说 object.key = object.key + 1 而是说: {$mergeObjects: [ "$$value", {key: {$add:["$$value.key",1]}} ]} */

,{$addFields: {Z: {$reduce: {
input: "$Z",
initialValue: {"A":0,"E":0,"D":0},
    in: {$cond: {
	if: {$eq:["$$this.v","active"]},
	then: {$mergeObjects: [ "$$value", {"A": {$add:["$$value.A",1]}} ]},
	else: {$cond: {
	    if: {$eq:["$$this.v","disabled"]},
	    then: {$mergeObjects: [ "$$value", {"D": {$add:["$$value.D",1]}} ]},
	    else: {$cond: {
		if: {$eq:["$$this.v","enabled"]},
		then: {$mergeObjects: [ "$$value", {"E": {$add:["$$value.E",1]}} ]},
		else: "$$value"
	    }}
	}}
    }}
}}
	  }}

// 将名称组合在一起并收集计数: ,{$group: { _id: "$name", X: {$push: "$Z"} }}

// 现在,再次运行$reduce以汇总计数并恢复大变量名称: ,{$project: {data: {$reduce: { input: "$X", initialValue: {"active":0,"enabled":0,"disabled":0}, in: {"active": {$add:["$$value.active","$$this.A"]}, "disabled": {$add:["$$value.disabled","$$this.D"]}, "enabled": {$add:["$$value.enabled","$$this.E"]} } }} }}

// 此时,我们"完成"了

<details> <summary>英文:</summary>

Here is a generalized solution:

db.foo.aggregate([
    {$project: {
        name: true,
    /*                                                                       
    We could get fancy with the output of $objectToArray, like               
    only going after statusOne and statusTwo:                                
                                                                             
       Z: {$filter: {                                                        
           input: {$objectToArray: &amp;amp;quot;$data&amp;amp;quot;},                                 
           cond: {$in: [&amp;amp;#39;$$this.k&amp;amp;#39;, [&amp;amp;#39;statusOne&amp;amp;#39;,&amp;amp;#39;statusTwo&amp;amp;#39;]]}              
           }}                                                                
                                                                             
       Or only looking for keys that start with &amp;amp;#39;status&amp;amp;#39;:                    
                                                                             
       Z: {$filter: {                                                        
           input: {$objectToArray: &amp;amp;quot;$data&amp;amp;quot;},                                 
           cond: {$eq: [&amp;amp;#39;status&amp;amp;#39;, {$substr:[&amp;amp;#39;$$this.k&amp;amp;#39;,0,6]}]}               
           }}                                                                
                                                                             
      Let&amp;amp;#39;s keep it simple now and take the whole thing                      
      without any filtering.                                                 
    */

    Z: {$objectToArray: &amp;amp;quot;$data&amp;amp;quot;}
}}
,{$unwind: &amp;amp;quot;$Z&amp;amp;quot;}

// Now we have docs like this:                                               
//{                                                                          
//  _id: ObjectId(&amp;amp;quot;64d651283d8bc34d3928366d&amp;amp;quot;),                               
//  name: &amp;amp;#39;another-test&amp;amp;#39;,                                                    
//  Z: {                                                                     
//    k: &amp;amp;#39;statusOne&amp;amp;#39;,                                                        
//    v: &amp;amp;#39;disabled&amp;amp;#39;                                                          
//  }                                                                        
//}                                                                          

// Now it is just a matter of grouping and reorganizing:                     
,{$group: {_id: {&amp;amp;quot;name&amp;amp;quot;:&amp;amp;quot;$name&amp;amp;quot;, &amp;amp;quot;v&amp;amp;quot;:&amp;amp;quot;$Z.v&amp;amp;quot;}, N:{$sum:1}}}
,{$group: {_id: &amp;amp;quot;$_id.name&amp;amp;quot;, data: {$push: {status:&amp;amp;quot;$_id.v&amp;amp;quot;, count:&amp;amp;quot;$N&amp;amp;quot;}} }}

]); </code></pre> <p>yields</p> <pre><code>{ _id: &amp;#39;test&amp;#39;, data: [ { status: &amp;#39;active&amp;#39;, count: 2 }, { status: &amp;#39;enabled&amp;#39;, count: 1 }, { status: &amp;#39;disabled&amp;#39;, count: 1 } ] } { _id: &amp;#39;another-test&amp;#39;, data: [ { status: &amp;#39;disabled&amp;#39;, count: 1 }, { status: &amp;#39;active&amp;#39;, count: 1 } ] } </code></pre> <h3>What if there are thousands of tests and we don't want to $unwind? {#what-if-there-are-thousands-of-tests-and-we-don39t-want-to-unwind}</h3> <p>You must be careful when throwing <code>$unwind</code> into a pipeline. If the average number of statuses per test gets large (say, 100) then there will be a LOT of docs in the pipeline. Below is an alternate solution that exploits <code>$reduce</code>:</p> <pre><code>c=db.foo.aggregate([ {$project: { name: true, Z: {$objectToArray: &amp;quot;$data&amp;quot;} }}

/*
 Instead of $unwind and using $group-$sum to count things, let&amp;amp;#39;s
 do it ourselves.  We will overwrite Z with a new Z.
 Also, since this is an iterim step, use shorter variable names for
 clarity.

 This if-then-else construction basically says:
     For each status type, status_type_count += 1
 This permits the input to have more than 1 status of the same type, e.g.

   &amp;amp;quot;name&amp;amp;quot;: &amp;amp;quot;test&amp;amp;quot;,
   &amp;amp;quot;data&amp;amp;quot;: {
     &amp;amp;quot;statusOne&amp;amp;quot;: &amp;amp;quot;enabled&amp;amp;quot;,
     &amp;amp;quot;statusTwo&amp;amp;quot;: &amp;amp;quot;enabled&amp;amp;quot;
   }

In MQL in a $reduce loop, we don&amp;#39;t say object.key = object.key + 1 Instead we say: {$mergeObjects: [ &amp;quot;$$value&amp;quot;, {key: {$add:[&amp;quot;$$value.key&amp;quot;,1]}} ]} */

,{$addFields: {Z: {$reduce: {
input: &amp;amp;quot;$Z&amp;amp;quot;,
initialValue: {&amp;amp;quot;A&amp;amp;quot;:0,&amp;amp;quot;E&amp;amp;quot;:0,&amp;amp;quot;D&amp;amp;quot;:0},
    in: {$cond: {
	if: {$eq:[&amp;amp;quot;$$this.v&amp;amp;quot;,&amp;amp;quot;active&amp;amp;quot;]},
	then: {$mergeObjects: [ &amp;amp;quot;$$value&amp;amp;quot;, {&amp;amp;quot;A&amp;amp;quot;: {$add:[&amp;amp;quot;$$value.A&amp;amp;quot;,1]}} ]},
	else: {$cond: {
	    if: {$eq:[&amp;amp;quot;$$this.v&amp;amp;quot;,&amp;amp;quot;disabled&amp;amp;quot;]},
	    then: {$mergeObjects: [ &amp;amp;quot;$$value&amp;amp;quot;, {&amp;amp;quot;D&amp;amp;quot;: {$add:[&amp;amp;quot;$$value.D&amp;amp;quot;,1]}} ]},
	    else: {$cond: {
		if: {$eq:[&amp;amp;quot;$$this.v&amp;amp;quot;,&amp;amp;quot;enabled&amp;amp;quot;]},
		then: {$mergeObjects: [ &amp;amp;quot;$$value&amp;amp;quot;, {&amp;amp;quot;E&amp;amp;quot;: {$add:[&amp;amp;quot;$$value.E&amp;amp;quot;,1]}} ]},
		else: &amp;amp;quot;$$value&amp;amp;quot;
	    }}
	}}
    }}
}}
	  }}


// Bring the names together and collect the counts:
,{$group: {
_id: &amp;amp;quot;$name&amp;amp;quot;, X: {$push: &amp;amp;quot;$Z&amp;amp;quot;}
}}

// Now, run a $reduce again to sum the counts AND put back the
// big variable names:
,{$project: {data: {$reduce: {
input: &amp;amp;quot;$X&amp;amp;quot;,
initialValue: {&amp;amp;quot;active&amp;amp;quot;:0,&amp;amp;quot;enabled&amp;amp;quot;:0,&amp;amp;quot;disabled&amp;amp;quot;:0},
    in: {&amp;amp;quot;active&amp;amp;quot;: {$add:[&amp;amp;quot;$$value.active&amp;amp;quot;,&amp;amp;quot;$$this.A&amp;amp;quot;]},
	 &amp;amp;quot;disabled&amp;amp;quot;: {$add:[&amp;amp;quot;$$value.disabled&amp;amp;quot;,&amp;amp;quot;$$this.D&amp;amp;quot;]},
	 &amp;amp;quot;enabled&amp;amp;quot;: {$add:[&amp;amp;quot;$$value.enabled&amp;amp;quot;,&amp;amp;quot;$$this.E&amp;amp;quot;]}
	}
}}
	}}


// At this point we are &amp;amp;quot;done&amp;amp;quot; information-wise but the OP was
// looking for an array of status as an RVAL not a key (e.g. &amp;amp;quot;A&amp;amp;quot;)
// so post-process:
,{$project: {
_id:0,
name:&amp;amp;quot;$_id&amp;amp;quot;,
data: {$map: {
    input: {$objectToArray: &amp;amp;quot;$data&amp;amp;quot;},
    in: {
	&amp;amp;quot;status&amp;amp;quot;:&amp;amp;quot;$$this.k&amp;amp;quot;,
	&amp;amp;quot;count&amp;amp;quot;:&amp;amp;quot;$$this.v&amp;amp;quot;,
    }
}}
}}

]); </code></pre> <br />

赞(4)
未经允许不得转载:工具盒子 » MongoDB聚合以多个字段分组。