使用MongoDB Java Driver中的数组字段进行项目

时间:2017-03-17 10:01:46

标签: java arrays mongodb mongodb-java

我收集如下。我通过用户查询此集合。

;; set difference
(into #{} (remove to-remove) query-result)
;; set intersection
(into #{} (filter to-keep) query-result)
;; set union
(into my-set query-result)

我想从集合中得到如下结果,我想在集合中获取关键字数组的最后一个元素

 {
      "user": "username",
      "sites": {
        "site": "abc",
        "keywords": [
          {
            "keyword": "keyword1",
            "dailyranks": [
              {
                "fild1": "value1"
              },
              {
                "fild2": "value2"
              },
              {
                "fild3": "value3"
              },
            ]
          },
          {
            "keyword": "keyword2",
            "dailyranks": [
              {
                "fild1": "value1"
              },
              {
                "fild2": "value2"
              },
              {
                "fild3": "value3"
              },  
            ]
          },
        ],
      }
    }

我使用 $ project 汇总,但没有成功。帮我解决这个问题。

使用的代码 -

[
    {
        "keyword" : "keyword1"
        "fild2" : "value2",
        "fild3" : "value3"
    },
    {
       "keyword" : "keyword2"
        "fild2" : "value2",
        "fild3" : "value3"
    },
]

1 个答案:

答案 0 :(得分:1)

您可以尝试以下聚合。

使用$map转换Keywords数组。

$map内,使用$arrayElemAt投影来自dailyranks$let运算符的最后和倒数第二个值来保存$arrayAtElem的结果并投影fild值。

db.coll.aggregate({
    $project: {
        keywords: {
            $map: {
                input: "$sites.keywords",
                as: "result",
                in: {
                    keyword: "$$result.keyword",
                    fild2: {$let: {vars: {obj: {$arrayElemAt: ["$$result.dailyranks", -2]}},in: "$$obj.fild2"}},
                    fild3: {$let: {vars: {obj: {$arrayElemAt: ["$$result.dailyranks", -1]}},in: "$$obj.fild3"}}
                }
            }
        }
    }
})

Java等效

MongoClient mongoClient = new MongoClient();
MongoDatabase db  = mongoClient.getDatabase("db")
MongoCollection<Document> collection = db.getCollection("collection");
List<Document> results =
    collection.aggregate(
        Arrays.asList(
            Aggregates.match(Filters.eq("user", modelLogin.getUSER_NAME() )),
            Aggregates.project(
                Projections.fields(
                    new Document("keywords",
                        new Document("$map",
                        new Document("input", "$sites.keywords").
                        append("as", "result").
                        append("in",
                            new Document("keyword","$$result.keyword").
                                append("fild2",
                                    new Document("$let",
                                    new Document("vars", new Document("obj",
                                            new Document("$arrayElemAt", Arrays.asList("$$result.dailyranks", -2)))).
                                    append("in", "$$obj.fild2" ))).
                                append("fild3",
                                    new Document("$let",
                                    new Document("vars", new Document("obj",
                                            new Document("$arrayElemAt", Arrays.asList("$$result.dailyranks", -1)))).
                                    append("in", "$$obj.fild3" )))
                            )
                        )
                    )))
        )).into(new ArrayList<>());