Mongoose - ObjectId
Mongoose 在创建 Schema 的时候默认配置一个 _id 属性,它的 SchemaType 是 mongoose.Schema.Types.ObjectId(等同于 mongoose.ObjectId)。
当创建一个 document 的时候会自动地生成一个 _id 属性,值是 mongoose.Types.ObjectId实例化的对象。
Mongoose 拒绝保存没有 _id 的文档。
查询到的 _id 是一个对象,可以调到它的 toString() 方法将其转化成字符串。
mongoose.Schema.Types.ObjectId (等同于 mongoose.ObjectId) 和 mongoose.Types.ObjectId 不同,前者仅仅是一个配置项,后者是构建 ObjectId 的类。
默认情况相当于这样手动实现:
const mongoose = require('mongoose')
const blogSchema = new mongoose.Schema({
_id: mongoose.Schema.Types.ObjectId // 配置类型
})
const blogModel = mongoose.model('Blog', blogSchema)
const blog = new blogModel({
_id: new mongoose.Types.ObjectId() // 手动生成 ObjectId
})
默认情况下,Mongoose 会自动管理(设置) _id 的值。如果是手动定义,要确保每个 document 都设置了的 _id。
Mongoose - 填充(Populate)
MongoDB 有一个类似 join 的聚合运算符 - $lookup。Mongoose 提供了一个更方便的方法 - populate(),用于在 document 中引用其它 collection 中的 document。
创建 Schema 时可以指定属性的 SchemaType 类型为 mongoose.Schema.Types.ObjectId(即存储引用模块的 _id),通过 ref 指定引用的哪个 Model,在填充时(调用 populate())Mongoose 会根据 _id 和 ref 查询对应的 Model 下的具体 document,将查询结果替换该属性的值。
const mongoose = require('mongoose')
const Schema = mongoose.Schema
const personSchema = new Schema({
_id: Schema.Types.ObjectId, // 手动指定 _id 的类型是 ObjectId
name: String,
age: Number,
stories: [
{
type: Schema.Types.ObjectId,
ref: 'Story' // 指定在填充时使用哪个 Model
}
] // stories 存储 ObjectIds 数组
})
const storySchema = Schema({
author: { type: Schema.Types.ObjectId, ref: 'Person' },
title: String,
fans: [{ type: Schema.Types.ObjectId, ref: 'Person' }]
})
const Story = mongoose.model('Story', storySchema)
const Person = mongoose.model('Person', personSchema)
存储引用(ref)
向 Schema.Types.ObjectId 类型的属性存值的时候,可以存储 ObjectId 对象或 <ObjectId>.toString() 返回的字符串,数据库会自动转化为 ObjectId 对象去存储。
const author = new Person({
// _id 建议由 Mongoose 默认生成
name: 'Ian Fleming',
age: 50
})
author.save(err => {
if (err) return handleError(err)
const story1 = new Story({
title: 'Casino Royale',
author: author._id // 存储引用的 `_id`
})
story1.save()
})
填充
在查询 story 时填充 person 信息:
Story
// .findOne() 返回一个 Query 对象
.findOne({title: 'Casino Royale'})
// Query.prototype.populate() 也返回 Query 对象,不接收回调函数
.populate('author')
// 需要调用 Query.prototype.exxc() 执行查询
.exec((err, story) => {
if (err) return handleError(err)
console.log(story.author.name) // Ian Fleming
})
原文链接:https://blog.youkuaiyun.com/u012961419/article/details/119994160