Mongo投影字段该字段不相同

时间:2017-03-26 03:55:52

标签: mongodb aggregation-framework

我的问题可能会引起混淆。让我进一步解释一下。

我有这样的汇总文件。

{
  "metric" : "user_act",
  "stream_id" : "f00001",
  "values" : {
    "likes" : 57,
    "comments" : 0,
    "shares" : 0
  }
}
{
    "metric" : "user_act",
    "stream_id" : "f00002",
    "values" : {
      "likes" : 28,
      "comments" : 0,
      "shares" : 1
    }
}

{
    "metric" : "user_act",
    "stream_id" : "t00001",
    "values" : {
      "favorites" : 5,
      "retweets" : 15
    }
}

我想通过总和喜欢,评论和分享来计算参与度。所以在计算之前,我必须在分组之前投影数据。如果没有任何数据集默认为0,我想将values.favorites映射到likesvalues.retweets映射到sharescomments

我尝试像下面这样的投影但不起作用,因为$ifNull的第二行的值会覆盖第一行。

{
  $project: {
    "stream_id" : 1, 
    "shares": {
      $ifNull: ["$values.retweets",0],
      $ifNull: ["$values.shares", 0]
    } ,
    "likes": {
      $ifNull: ["$values.favorites",0],
      $ifNull: ["$values.likes", 0]
    } ,
    "comments": {
      $ifNull: ["$values.replys",0],
      $ifNull: ["$values.comments", 0]
    }
  }
}

任何人都有任何想法?先谢谢你。

[更新] 我尝试像这样进行投影但不能在case:中工作如何检查该字段是否存在?

{
  $project: {
    "stream_id" : 1,
      "shares": {
        $switch: {
          branches: [
            { case: {$ne: ["$values.retweets", 0]},
              then: {$ifNull: ["$values.retweets", "$values.retweets"]}
            },
            { case: {$ne: ["$values.shares", 0]},
              then: {$ifNull: ["$values.shares", "$values.shares"]}
            }
          ],
          default: 0
        }
      }
  }
}

1 个答案:

答案 0 :(得分:1)

您可以尝试$cond投影。

db.collection.aggregate({
  $project: {
    "stream_id" : 1, 
    "shares": { $cond: [ { $eq:[ { $ifNull: [ "$values.shares", 0 ] }, 0 ] },{ $ifNull: [ "$values.retweets", 0 ] }, "$values.shares" ] },
    "likes": { $cond: [ { $eq:[ { $ifNull: [ "$values.likes", 0 ] }, 0 ] }, { $ifNull: [ "$values.favorites", 0 ] }, "$values.likes" ] },
    "comments": { $cond: [ { $eq:[ { $ifNull: [ "$values.comments", 0] }, 0 ] }, { $ifNull: [ "$values.replys", 0 ] }, "$values.comments" ] }
}})

使用$switch

db.collection.aggregate([{
  $project: {
    "stream_id" : 1,
      "shares": {
        $switch: {
          branches: [
            { case: { $ifNull: [ "$values.shares", false ] },
              then: "$values.shares"
            },
           { case: { $ifNull: [ "$values.retweets", false ] },
              then: "$values.retweets"
            }],
          default: 0
        }
      }
  }
}])