sql >> Databasteknik >  >> NoSQL >> MongoDB

Mongo Query kapslade fältvärden med okända föräldernycklar på två nivåer

Map reduce låter dig bearbeta namngivna nycklar men aggregering är vägen att gå för effektiva frågor.

Du måste modellera data som en rad inbäddade dokument för aggregeringsramverk.

Jag har gett dig två alternativ. Du kan testa dem för din datauppsättning och se vilken som fungerar bäst för dig.

Något liknande

"v":[
  {
    "minute":1,
    "seconds":[
      {
        "second":54,
        "data":{
         "field1":7.373158,
         "entry_id":4635,
         "field3":0.19,
         "field2":88
       }
      }
    ]
  },
  {
    "minute":2,
    "seconds":...
  }
]

Nu kan du enkelt fråga efter objekt som har sensoravläsning:"field1">2.

db.col.aggregate(
  [{"$match":{"v.seconds.data.field1":{"$gt":2}}},
   {"$unwind":"$v"}, 
   {"$match":{"v.seconds.data.field1":{"$gt":2}}},
   {"$unwind":"$v.seconds"}, 
   {"$match":{"v.seconds.data.field1":{"$gt":2}}},
   {"$project":{"data":"$v.seconds.data"}}]
)

Alternativt kan du dela upp dokumenten per minut. Något liknande

"v":[
  {
    "second":1,
    "data":{
       "field1":7.373158,
       "entry_id":4635,
       "field3":0.19,
       "field2":88
     }
  },
  {
     "second":2,
     "data":...
  }
]

Du kan nu fråga som (med index på v.data.field1)

db.col.aggregate(
  [{"$match":{"v.data.field1":{"$gt":2}}},
   {"$unwind":"$v"}, 
   {"$match":{"v.data.field1":{"$gt":2}}},
   {"$project":{"data":"$v.data"}}]
)

Du kan fråga objekt som har sensoravläsning:"field1">2 och "field3">5

Använder första strukturen

db.col.aggregate(
  [{"$match":{"v":{"$elemMatch":{"seconds": {$elemMatch:{"field1":{$gt":2},"field3":{$gt":5}}}}}}},
  {"$unwind":"$v"}, 
    {"$match":{"v.seconds": {$elemMatch:{"field1":{$gt":2},"field3":{$gt":5}}}}},
  {"$unwind":"$v.seconds"}, 
  {"$project":{"data":"$v.seconds.data"}}]
)

Använder den andra strukturen

db.col.aggregate(
  [{"$match":{"v.data":{$elemMatch:{"field1":{$gt":2},"field3":{$gt":5}}}}},
  {"$unwind":"$v"}, 
  {"$match":{"v.data.field1":{"$gt":2},"v.data.field3":{"$gt":5} }},
  {"$project":{"data":"$v.data"}}]
)

Mongo Update 3.6

$match med $expr som accepterar aggregeringsuttryck.

$gt > 0 - aggregeringsuttryck för att kontrollera var summan av alla matchande sekunders kriterier i en minut är större än 0

$objectToArray för att konvertera de namngivna nycklarna till nyckelvärdespar följt av $filter sekunder på inmatningskriterier och utgångsnummer för matchande sekundpost.

db.testcol.aggregate(
{"$match":{
  "$expr":{
    "$gt":[
      {"$sum":{
        "$map":{
          "input":{"$objectToArray":"$v"},
          "as":"secondsofminute",
          "in":{
            "$size":{
              "$filter":{
                "input":{"$objectToArray":"$$secondsofminute.v"},
                "as":"seconds",
                "cond":{"$gt":["$$seconds.v.field2",2]}
              }
            }
          }
        }
      }},
    0]
  }
}})

Mongo Update 3.4 - Ersätt $expr med $redact

db.col.aggregate(
 {"$redact":{
  "$cond":{
    "if":{
      "$gt":[
        {"$sum":{
          "$map":{
            "input":{"$objectToArray":"$v"},
            "as":"secondsofminute",
            "in":{
              "$size":{
                "$filter":{
                  "input":{"$objectToArray":"$$secondsofminute.v"},
                  "as":"seconds",
                  "cond":{"$gt":["$$seconds.v.field2",2]}
                }
              }
            }
          }
        }},
        0]
    },
   "then":"$$KEEP",
   "else":"$$PRUNE"
  }
}})



  1. MongoDB sortering vs aggregerad $sort på arrayindex

  2. Databasstorleken på disken ökar när en multipel av CSV-filen jag mongoimporterar?

  3. MongoError:drivrutinen är inkompatibel med denna serverversion

  4. Snabba upp MongoDB-aggregation