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

Interroger le tableau MongoDB et trier avec les éléments les plus correspondants

Vous pouvez le faire avec le cadre d'agrégation, même si ce n'est pas facile. Le problème réside dans le fait qu'il n'y a pas de $in opérateur dans le cadre du cadre d'agrégation. Vous devez donc faire correspondre par programmation chacun des éléments du tableau, ce qui devient très compliqué. modifier  :réorganisé de sorte que la correspondance soit la première, au cas où $in vous aide à filtrer une bonne partie.

db.test.aggregate(
  {$match:{"array.1":{$in:[100, 140,80]}}}, // filter to the ones that match
  {$unwind:"$array.1"}, // unwinds the array so we can match the items individually
  {$group: { // groups the array back, but adds a count for the number of matches
    _id:"$_id", 
    matches:{
      $sum:{
        $cond:[
          {$eq:["$array.1", 100]}, 
          1, 
          {$cond:[
            {$eq:["$array.1", 140]}, 
            1, 
            {$cond:[
              {$eq:["$array.1", 80]}, 
              1, 
              0
              ]
            }
            ]
          }
          ]
        }
      }, 
    item:{$first:"$array.item"}, 
    "1":{$push:"$array.1"}
    }
  }, 
  {$sort:{matches:-1}}, // sorts by the number of matches descending
  {$project:{matches:1, array:{item:"$item", 1:"$1"}}} // rebuilds the original structure
);

sorties :

{
"result" : [
    {
        "_id" : ObjectId("50614c02162d92b4fbfa4448"),
        "matches" : 2,
        "array" : {
            "item" : 3,
            "1" : [
                100,
                90,
                140
            ]
        }
    },
    {
        "_id" : ObjectId("50614bb2162d92b4fbfa4446"),
        "matches" : 1,
        "array" : {
            "item" : 1,
            "1" : [
                100,
                130,
                255
            ]
        }
    }
],
"ok" : 1
}

Vous pouvez laisser les matches champ hors du résultat si vous le laissez hors du $project à la fin.