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

Kartezjański produkt C# Mongodb dokumentów z wieloma tablicami obiektów

Możesz spróbować poniżej potoku agregacji.

Uwaga mergeObjects operator agregacji jest dostępny w 3.5.6 + wersja rozwojowa, która zostanie wprowadzona do nadchodzącego 3.6 zwolnienie.

db.collection.find();
{
 "data" : [
  [
   {
    "movie" : "starwars",
    "showday" : "monday"
   },
   {
    "movie" : "batman",
    "showday" : "thursday"
   },
   {
    "movie" : "sleepless",
    "showday" : "tuesday"
   }
  ],
  [
   {
    "actor" : "angelina",
    "location" : "new york"
   },
   {
    "actor" : "jamie",
    "location" : "california"
   },
   {
    "actor" : "mcavoy",
    "location" : "arizona"
   }
  ]
 ]
}

Agregacja przy użyciu wyrażenia warunkowego.

aggregate({
 $project: {
  cp: {
   $reduce: {
    input: "$data",
    initialValue: {
     $arrayElemAt: ["$data", 0] // Set the initial value to the first element of the arrays.
    },
    in: {
     $let: {
      vars: {
       currentr: "$$this", // Current processing element
       currenta: "$$value" // Current accumulated value 
      },
      in: {
       $cond: [{ // Conditional expression to return the accumulated value as initial value for first element
        $eq: ["$$currentr", "$$currenta"]
       },
       "$$currenta",
       { // From second element onwards prepare the cartesian product
        $reduce: {
         input: {
          $map: {
           input: "$$currenta",
           as: a"a",
           in: {
            $map: {
             input: "$$currentr",
             as: r"r",
             in: {
              $mergeObjects: ["$$a", "$$r"] // Merge accumulated value with the current processing element
             }
            }
           }
          }
         },
         initialValue: [],
         in: {
         $concatArrays: ["$$value", "$$this"] // Reduce the merged values which will be used as accumulator for next element
         }
        }
       }]
      }
     }
    }
   }
  }
 }
});

Agregacja (przy użyciu $setUnion ).

To rozwiązanie zostało dodane tylko w celu pominięcia wyrażenia warunkowego, aby zapewnić bardziej czytelny potok.

aggregate({
 $project: {
  cp: {
   $reduce: {
    input: "$data",
    initialValue: {
     $arrayElemAt: ["$data", 0] // Set the initial value to the first element of the arrays.
    },
    in: {
     $let: {
      vars: {
       currentr: "$$this", // Current processing element
       currenta: "$$value" // Current accumulated value 
      },
      in:{ 
       $reduce: {
        input: {
         $map: {
          input: "$$currenta",
          as: "a",
          in: {
           $map: {
            input: "$$currentr",
            as: "r",
            in: {
             $mergeObjects: ["$$a", "$$r"] // Merge accumulated value with the current processing element
            }
           }
          }
         }
        },
        initialValue: [],
        in: {
         $setUnion: ["$$value", "$$this"] // Reduce the merged values which will be used as accumulator for next element
        }
       }
      }
     }
    }
   }
  }
 }
});

Aktualizacja

Oba powyższe rozwiązania nie będą działać z powtarzającymi się wartościami w tablicach, jak zauważono w komentarzach Asyi Kamsky poniżej, z powodu nieprawidłowego $cond w pierwszym rozwiązaniu i $setUnion w drugim rozwiązaniu.

Prawidłowe rozwiązanie to

zacznij od initialValue z [ { } ]

Lub

zmień input aby wykluczyć pierwszy element, taki jak input: {$slice:["$data", 1, {$subtract:[{$size:"$data"},1]}]},

Kompletny potok agregacji

aggregate({
 $project: {
  cp: {
   $reduce: {
    input: {$slice:["$data", 1, {$subtract:[{$size:"$data"},1]}]},
    initialValue: {$arrayElemAt:["$data",0]},
    in: {
     $let: {
      vars: {
       currentr: "$$this", 
       currenta: "$$value" 
      },
      in:{ 
       $reduce: {
        input: {
         $map: {
          input: "$$currenta",
          as: "a",
          in: {
           $map: {
            input: "$$currentr",
            as: "r",
            in: {
             $mergeObjects: ["$$a", "$$r"] 
            }
           }
          }
         }
        },
        initialValue: [],
        in: {
         $concatArrays: ["$$value", "$$this"] 
        }
       }
      }
     }
    }
   }
  }
 }
});

Odniesienie:Iloczyn kartezjański wielu tablic w JavaScript




  1. Redis
  2.   
  3. MongoDB
  4.   
  5. Memcached
  6.   
  7. HBase
  8.   
  9. CouchDB
  1. Jak ograniczyć usuwanie w MongoDB w celu zbierania relacji

  2. Dodaj pole do istniejącego dokumentu MongoDB (z Mongoose w Node.js)

  3. Czy meteor automatycznie aktualizuje wersję mongodb?

  4. W MongoDB, jeśli kolekcja jest usuwana, indeksy są również usuwane automatycznie?

  5. Nie można uruchomić robomongo.sh przy użyciu sh robomongo.sh w Ubuntu 14.04