How to sort sub-documents in the array field?

2019-01-29 02:14发布

I'm using the MongoDB shell to fetch some results, ordered. Here's a sampler,

{
"_id" : "32022",
"topics" : [
    {
        "weight" : 281.58551703724993,
        "words" : "some words"
    },
    {
        "weight" : 286.6695125796183,
        "words" : "some more words"
    },
    {
        "weight" : 289.8354232846977,
        "words" : "wowz even more wordz"
    },
    {
        "weight" : 305.70093587160807,
        "words" : "WORDZ"
    }]
}

what I want to get is, same structure, but ordered by "topics" : []

{
"_id" : "32022",
"topics" : [
    {
        "weight" : 305.70093587160807,
        "words" : "WORDZ"
    },
    {
        "weight" : 289.8354232846977,
        "words" : "wowz even more wordz"
    },
    {
        "weight" : 286.6695125796183,
        "words" : "some more words"
    },
    {
        "weight" : 281.58551703724993,
        "words" : "some words"
    },
    ]
}

I managed to get some ordered results, but no luck in grouping them by id field. is there a way to do this?

3条回答
欢心
2楼-- · 2019-01-29 02:52

If you don't want to update but only get documents, you can use the following query

 db.test.aggregate(
 [
   {$unwind : "$topics"},
   {$sort : {"topics.weight":-1}},
   {"$group": {"_id": "$_id", "topics": {"$push": "$topics"}}}
 ]
)
查看更多
不美不萌又怎样
3楼-- · 2019-01-29 03:03

MongoDB doesn't provide a way to do this out of the box but there is a workaround which is to update your documents and use the $sort update operator to sort your array.

db.collection.update_many({}, {"$push": {"topics": {"$each": [], "$sort": {"weight": -1}}}})

You can still use the .aggregate() method like this:

db.collection.aggregate([
    {"$unwind": "$topics"}, 
    {"$sort": {"_id": 1, "topics.weight": -1}}, 
    {"$group": {"_id": "$_id", "topics": {"$push": "$topics"}}}
])

But this is less efficient if all you want is sort your array, and you definitely shouldn't do that.


You could always do this client side using the .sort or sorted function.

查看更多
戒情不戒烟
4楼-- · 2019-01-29 03:10

It works for me:

db.getCollection('mycollection').aggregate(
{$project:{topics:1}},
{$unwind:"$topics"},
{$sort :{"topics.words":1}})
查看更多
登录 后发表回答