MongoDB
 sql >> Baza danych >  >> NoSQL >> MongoDB

Wyszukiwanie z tablicą obiektów

Zasadniczo musisz $unwind najpierw tablica. MongoDB nie może jeszcze pracować z właściwością „inner” obiektu w tablicy jako źródłem dla $lookup .

Również dla wydajności naprawdę powinniśmy użyć $concatArrays najpierw „dołącz” do źródła tablicy, a potem wykonaj tylko jeden $lookup operacja:

Project.aggregate([
  { "$match": { "project_id": projectId} },
  { "$project": {
    "project_id": 1,
    "updated_at": 1,
    "created_at": 1,
    "owner": 1,
    "name": 1,
    "combined": {
      "$concatArrays": [
        { "$map": {
          "input": {
            "$filter": {
              "input": "$uploaded_files",
              "as": "uf",
              "cond": { "$eq": ["$$uf.upload_id", uploadId ] }
            }
          },
          "as": "uf",
          "in": {
            "$arrayToObject": {
              "$concatArrays": [
                { "$objectToArray": "$$uf" },
                [{ "k": "type", "v": "uploaded_files" }]
              ]
            }
          }
        }},
        { "$map": {
          "input": {
            "$filter": {
              "input": "$file_history",
              "as": "fh",
              "cond": { "$eq": ["$$fh.upload_id", uploadId ] }
            }
          },
          "as": "fh",
          "in": {
            "$arrayToObject": {
              "$concatArrays": [
                { "$objectToArray": "$$fh" },
                [{ "k": "type", "v": "file_history" }]
              ]
            }
          }
        }}
      ]
    }
  }},
  { "$unwind": "$combined" },
  { "$lookup": {
    "from": "files",
    "localField": "combined.file",
    "foreignField": "_id",
    "as": "combined.file"
  }},
  { "$unwind": "$combined.file" },
  { "$lookup": {
    "from": "users",
    "localField": "owner",
    "foreignField": "_id",
    "as": "owner"
  }},
  { "$unwind": "$owner" },
  { "$group": {
    "_id": "$_id",
    "project_id": { "$first": "$project_id" },
    "updated_at": { "$first": "$updated_at" },
    "created_at": { "$first": "$created_at" },
    "owner": { "$first": "$owner" },
    "name": { "$first": "$name" },
    "combined": { "$push": "$combined" }
  }},
  { "$project": {
    "project_id": 1,
    "updated_at": 1,
    "created_at": 1,
    "owner": 1,
    "name": 1,
    "uploaded_files": {
      "$filter": {
        "input": "$combined",
        "as": "cf",
        "cond": { "$eq": [ "$$cf.type", "uploaded_files" ] }
      }    
    },
    "file_history": {
      "$filter": {
        "input": "$combined",
        "as": "cf",
        "cond": { "$eq": [ "$$cf.type", "file_history" ] }
      }    
    }
  }}
])

W skrócie

  1. Połącz dwie tablice w źródle i oznacz je, a następnie $unwind najpierw

  2. Wykonaj $lookup na połączonych szczegółach i $unwind to

  3. Wykonaj $lookup na innym zagranicznym źródle i $unwind to

  4. $group dokument z powrotem razem z pojedynczą tablicą.

  5. $filter przez pole "nazwy znaczników" lub "typ" dodaliśmy, aby "oddzielić" tablice.

Możesz wykonać ten sam proces, używając po prostu $unwind na każdej tablicy, a następnie wykonując "join" i ponownie grupując. Ale tak naprawdę wymaga to znacznie więcej kroków, niż po prostu „łączenia” w pierwszej kolejności.




  1. Redis
  2.   
  3. MongoDB
  4.   
  5. Memcached
  6.   
  7. HBase
  8.   
  9. CouchDB
  1. Zmiana hasła w MongoDB dla istniejącego użytkownika

  2. Kiedy dokument został dodany do kolekcji MongoDB?

  3. Używanie $slice z $regex razem w tablicy subDocument w mongodb

  4. Integracja ClusterControl z SNMP — dowód koncepcji:część pierwsza

  5. Nie można połączyć się z mongodb Błąd:nie można połączyć się z serwerem 127.0.0.1:27017 w src/mongo/shell/mongo.js:L112