如何将两个数组转换为 mongoDB 中的对象,其中第一个数组具有多个相同的值

Vee*_*Vee 5 mongodb mongodb-query mongodb-compass

在使用 mongo 聚合进行一些处理后,我有一个类似的集合:

[{
 field1: 10,
 field2: 50,
 field3: { 
           name: ["a","b","a","a"], 
           value: [1,2,3,4]
         }
},...]
Run Code Online (Sandbox Code Playgroud)

我正在努力寻找一种方法将其转换为:

[{
 field1: 10,
 field2: 50,
 a:[1,3,4],
 b:[2]
},...]
Run Code Online (Sandbox Code Playgroud)

在 mongoshell 或指南针中使用 mongo 聚合

Pur*_*jax 1

所以我们的想法是:

  1. field3从say中选择一个数组name来生成数组 range [0, 1, 2, 3]
  2. 同时循环遍历namevaluearray 以获取{ k: "", v: "" }.
  3. $unwind用于分组的前一阶段的数组。
  4. 按键分组field3.k以累积所有值field3.v
  5. 再次 group bynull生成模式数组{ k: "", v: "" }
  6. field1和连接field2到前一阶段的数组中。
  7. 最后使用将数组转换为对象$arrayToObject

尝试这个:

db.testCollection.aggregate([
    {
        $addFields: {
            field3: {
                $map: {
                    input: { $range: [0, { $size: "$field3.name" }] },
                    as: "index",
                    in: {
                        k: { $arrayElemAt: ["$field3.name", "$$index"] },
                        v: { $arrayElemAt: ["$field3.value", "$$index"] }
                    }
                }
            }
        }
    },
    { $unwind: "$field3" },
    {
        $group: {
            _id: "$field3.k",
            field1: { $first: "$field1" },
            field2: { $first: "$field2" },
            k: { $first: "$field3.k" },
            v: { $push: "$field3.v" }
        }
    },
    {
        $group: {
            _id: null,
            field1: { $first: "$field1" },
            field2: { $first: "$field2" },
            array: {
                $push: { k: "$k", v: "$v" }
            }
        }
    },
    {
        $addFields: {
            array: {
                $concatArrays: [
                    [{ k: "field1", v: "$field1" }, { k: "field2", v: "$field2" }],
                    "$array"
                ]
            }
        }
    },
    {
        $replaceRoot: {
            newRoot: { $arrayToObject: "$array" }
        }
    }
]);
Run Code Online (Sandbox Code Playgroud)

输出

{
    "field1" : 10,
    "field2" : 50,
    "a" : [
        1,
        3,
        4
    ],
    "b" : [
        2
    ]
}
Run Code Online (Sandbox Code Playgroud)