我使用Mongoose.js并且无法解决3级层次结构文档的问题。
有两种方法可以做到。
首先 - 没有参考。
C = new Schema({
'title': String,
});
B = new Schema({
'title': String,
'c': [C]
});
A = new Schema({
'title': String,
'b': [B]
});
我需要显示C记录。我怎么能填充/找到它,只知道C的_id?
我尝试使用:
A.findOne({'b.c._id': req.params.c_id}, function(err, a){
console.log(a);
});
但我不知道如何从者回归到达我需要的对象c对象。
第二如果使用refs:
C = new Schema({
'title': String,
});
B = new Schema({
'title': String,
'c': [{ type: Schema.Types.ObjectId, ref: 'C' }]
});
A = new Schema({
'title': String,
'b': [{ type: Schema.Types.ObjectId, ref: 'B' }]
});
如何填充所有B,C记录以获取层次结构?
我试着用这样的东西:
A
.find({})
.populate('b')
.populate('b.c')
.exec(function(err, a){
a.forEach(function(single_a){
console.log('- ' + single_a.title);
single_a.b.forEach(function(single_b){
console.log('-- ' + single_b.title);
single_b.c.forEach(function(single_c){
console.log('--- ' + single_c.title);
});
});
});
});
但是它将为single_c.title返回undefined。我有办法填充它吗?
感谢。
答案 0 :(得分:42)
自Mongoose 3.6起添加了the ability to recursively populate related documents。以下是您可能会这样做的示例:
UserList.findById(listId)
.populate('refUserListItems')
.exec(function(err, doc){
UserListItem.populate(doc.refUserListItems, {path:'refSuggestion'},
function(err, data){
console.log("User List data: %j", doc);
cb(null, doc);
}
);
});
在这种情况下,我使用引用的文档填充'refUserListItems'中的id数组。然后查询的结果将传递到另一个填充查询,该查询引用我想要填充的原始填充文档的字段 - 'refSuggestion'。
注意第二个(内部)填充 - 这是神奇发生的地方。您可以继续嵌套这些填充并添加越来越多的文档,直到您以所需的方式构建图形。
需要一点时间来消化它是如何工作的,但是如果你通过它,它是有道理的。
答案 1 :(得分:29)
在Mongoose 4中,您可以像这样填充多级(即使在不同的数据库或实例中)
A
.find({})
.populate({
path: 'b',
model: 'B',
populate: {
path: 'c',
model: 'C'
}
})
.exec(function(err, a){});
答案 2 :(得分:28)
在Mongoose 4中,您可以跨多个级别填充文档:
假设您有用户架构,可以跟踪用户的朋友。
var userSchema = new Schema({
name: String,
friends: [{ type: ObjectId, ref: 'User' }]
});
首先,populate()
可让您获得用户朋友列表。但是,如果您还想要用户的朋友朋友呢?在这种情况下,您可以指定populate
选项告诉mongoose填充所有用户朋友的friends
数组:
User.
findOne({ name: 'Val' }).
populate({
path: 'friends',
// Get friends of friends - populate the 'friends' array for every friend
populate: { path: 'friends' }
});
答案 3 :(得分:4)
我迟到了,但我写了Mongoose plugin,这使得执行深度模型人口非常简单。对于您的示例,您可以执行此操作以填充b
和c
:
A.find({}, function (err, docs) {
A.deepPopulate(docs, 'b.c', cb)
}
您还可以为每个填充路径指定Mongoose populate options,如下所示:
A.deepPopulate(docs, 'b.c', {
b: {
select: 'name'
}
}, cb)
查看插件documentation以获取更多信息。