Как присоединиться к запросу в mongodb? - PullRequest
32 голосов
/ 30 декабря 2010

У меня есть коллекция пользовательских документов, подобная этой:

User {
   id:"001"
   name:"John",
   age:30,
   friends:["userId1","userId2","userId3"....]
}

У пользователя много друзей, у меня следующий запрос в SQL:

select * from user where in (select friends from user where id=?) order by age

Я хотел бы иметь что-то похожее в MongoDB.

Ответы [ 9 ]

31 голосов
/ 17 мая 2016

Чтобы получить все с помощью одного запроса с использованием функции $ lookup структуры агрегации, попробуйте следующее:

db.User.aggregate(
    [
        // First step is to extract the "friends" field to work with the values
        {
            $unwind: "$friends"
        },
        // Lookup all the linked friends from the User collection
        {
            $lookup:
            {
                from: "User",
                localField: "friends",
                foreignField: "_id",
                as: "friendsData"
            }
        },
        // Sort the results by age
        {
            $sort: { 'friendsData.age': 1 }
        },
        // Get the results into a single array
        {
            $unwind: "$friendsData"
        },
        // Group the friends by user id
        {
            $group:
            {
                _id: "$_id",
                friends: { $push: "$friends" },
                friendsData: { $push: "$friendsData" }
            }
        }
    ]
)

Допустим, содержимое вашей коллекции пользователей выглядит следующим образом:

{
    "_id" : ObjectId("573b09e6322304d5e7c6256e"),
    "name" : "John",
    "age" : 30,
    "friends" : [
        "userId1",
        "userId2",
        "userId3"
    ]
}
{ "_id" : "userId1", "name" : "Derek", "age" : 34 }
{ "_id" : "userId2", "name" : "Homer", "age" : 44 }
{ "_id" : "userId3", "name" : "Bobby", "age" : 12 }

Результат запроса будет:

{
    "_id" : ObjectId("573b09e6322304d5e7c6256e"),
    "friends" : [
        "userId3",
        "userId1",
        "userId2"
    ],
    "friendsData" : [
        {
            "_id" : "userId3",
            "name" : "Bobby",
            "age" : 12
        },
        {
            "_id" : "userId1",
            "name" : "Derek",
            "age" : 34
        },
        {
            "_id" : "userId2",
            "name" : "Homer",
            "age" : 44
        }
    ]
}
30 голосов
/ 04 января 2011

Редактировать: этот ответ применим только к версиям MongoDb до версии 3.2.

Вы не можете делать то, что хотите, только одним запросом. Сначала нужно получить список идентификаторов друзей, а затем передать эти идентификаторы во второй запрос, чтобы получить документы и отсортировать их по возрасту.

var user = db.user.findOne({"id" : "001"}, {"friends": 1})
db.user.find( {"id" : {$in : user.friends }}).sort("age" : 1);
10 голосов
/ 25 января 2016

https://docs.mongodb.org/manual/reference/operator/aggregation/lookup/

Это документ для запроса соединения в mongodb, это новая функция из версии 3.2.

Так что это будет полезно.

5 голосов
/ 30 декабря 2010

MongoDB не имеет объединений, но в вашем случае вы можете сделать:

db.coll.find({friends: userId}).sort({age: -1})
2 голосов
/ 19 октября 2014

один вид присоединения к запросу в mongoDB, это запрос в одной коллекции идентификаторов, которые соответствуют, помещают идентификаторы в список (список ID) и находят использование в другой (или той же) коллекции с $ in: idlist

u = db.friends.find({"friends": ? }).toArray()
idlist= []
u.forEach(function(myDoc) { idlist.push(myDoc.id ); } )
db.friends.find({"id": {$in : idlist} } )
0 голосов
/ 23 июля 2018

Заполняйте только массив друзей.

User.findOne({ _id: "userId"})
.populate('friends')
.exec((err, user) => {
    //do something
});

Результат такой же, как этот:

{
    "_id" : "userId",
    "name" : "John",
    "age" : 30,
    "friends" : [
        { "_id" : "userId1", "name" : "Derek", "age" : 34 }
        { "_id" : "userId2", "name" : "Homer", "age" : 44 }
        { "_id" : "userId3", "name" : "Bobby", "age" : 12 }
    ]
}

То же самое: Mongoose - использование Заполнить массив массива ObjectId

0 голосов
/ 13 декабря 2017

Вы можете сделать это за один раз, используя mongo-join-query. Вот как это будет выглядеть:

const joinQuery = require("mongo-join-query");

joinQuery(
    mongoose.models.User,
    {
        find: {},
        populate: ["friends"],
        sort: { age: 1 },
    },
    (err, res) => (err ? console.log("Error:", err) : console.log("Success:", res.results))
);

В результате ваши пользователи будут упорядочены по возрасту, а все объекты друзей встроены.

Как это работает?

За кулисами mongo-join-query будет использовать вашу схему Mongoose, чтобы определить, к каким моделям присоединиться, и создаст конвейер агрегации , который будет выполнять объединение и запрос.

0 голосов
/ 28 ноября 2014
var p = db.sample1.find().limit(2) , 
    h = [];
for (var i = 0; i < p.length(); i++) 
{
  h.push(p[i]['name']);
}
db.sample2.find( { 'doc_name': { $in : h } } ); 

у меня это работает.

0 голосов
/ 07 сентября 2012

Вы можете использовать playOrm, чтобы делать то, что вы хотите, в одном запросе (с масштабируемым SQL S-SQL).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...