mongoose 的 document 与MongoDB 的 document 一一对应。每个document都是它的model的实例。
在MongoDB中有很多方法检索document。我们不会在这一节中涉及。详细请看 querying 节。
有很多种方法修改 document。我们首先看传统方法findById。
Tank.findById(id, function (err, tank) { if (err) return handleError(err); tank.size = 'large'; tank.save(function (err) { if (err) return handleError(err); res.send(tank); }); });
这种方法包括先从Mongo检索document,然后发出修改命令(通过调用保存触发)。可是,如果我们不需要在应用程序中返回document而是只不过想直接修改数据库中的属性,Model#update 适合我们。
Tank.update({ _id: id }, { $set: { size: 'large' }}, callback);
如果我们想要应用程序中返回document,有一个更好的选项
Tank.findByIdAndUpdate(id, { $set: { size: 'large' }}, function (err, tank) { if (err) return handleError(err); res.send(tank); });
静态方法findAndUpdate/Remove 最多只改变一个document,并只通过一个调用返回到数据库。在 findAndModifu 主题 有 几 种 变化。阅读API文档查看更多细节。注意findAndUpdate/Remove在修改数据库前不执行任何钩子或验证。如果你想要钩子和验证,先检索 document 然后保存它。
document在保存前进行验证。详情阅读API文档或validation节。
sub-document是各自有着schema的document,是一个父元素数组的元素。
var childSchema = new Schema({ name: 'string' }); var parentSchema = new Schema({ children: [childSchema] })
sub-document享有所有与普通document相同的特征。唯一不同的是它们不单独保存,当它们的顶层父document保存时它们才被保存。
var Parent = mongoose.model('Parent', parentSchema); var parent = new Parent({ children: [{ name: 'Matt' }, { name: 'Sarah' }] }) parent.children[0].name = 'Matthew'; parent.save(callback);
如果在子文档中间件发生错误,它冒泡到父的save()回调,因此错误处理是小事一桩。
childSchema.pre('save', function (next) { if ('invalid' == this.name) return next(new Error('#sadpanda')); next(); }); var parent = new Parent({ children: [{ name: 'invalid' }] }); parent.save(function (err) { console.log(err.message) // #sadpanda })
每一个document都有_id,.DocumentArrays有特殊的 id 方法来通过_id来查找document。
var doc = parent.children.id(id);
mongoose数组方法如push、unshift、addToSet等将参数显式转换成恰当的类型。
var Parent = mongoose.model('Parent'); var parent = new Parent; // create a comment parent.children.push({ name: 'Liesl' }); var subdoc = parent.children[0]; console.log(subdoc) // { _id: '501d86090d371bab2c0341c5', name: 'Liesl' } subdoc.isNew; // true parent.save(function (err) { if (err) return handleError(err) console.log('Success!'); });
子文档的创建不需通过使用创建方法MongooseArrays添加到数组。
var newdoc = parent.children.create({ name: 'Aaron' });
每个子文档都有自己的删除方法:
var doc = parent.children.id(id).remove(); parent.save(function (err) { if (err) return handleError(err); console.log('the sub-doc was removed') });
如果你不需要访问的子文档schena的实例,你也可以通过简单传递一个对象字面量声明子文档。
var parentSchema = new Schema({ children: [{ name: 'string' }] })
New in 4.2.0
你还可以在不使用数组的方式下嵌入schema。
var childSchema = new Schema({ name: 'string' }); var parentSchema = new Schema({ child: childSchema });
一个嵌入式子文档的行为类似于一个嵌入式数组。当它的父文档保存时它才保存、它的前/后document中间件被执行。
childSchema.pre('save', function(next) { console.log(this.name); // prints 'Leia' }); var Parent = mongoose.model('Parent', parentSchema); var parent = new Parent({ child: { name: 'Luke' } }) parent.child.name = 'Leia'; parent.save(callback); // Triggers the pre middleware.
你可以给schema的某些path定义默认值。如果你创建的新document没有设置那个path,默认值会取代。
var schema = new Schema({ name: String, role: {type: String, default: 'guitarist'} }); var Person = db.model('Person', schema); var axl = new Person({name: 'Axl Rose', role: 'singer'}); assert.equal(axl.role, 'singer'); var slash = new Person({name: 'Slash'}); assert.equal(slash.role, 'guitarist');
你还可以将默认schema选项设置为一个函数。mongoose 将执行那个函数并使用它的返回值作为默认值。
var schema = new Schema({ title: String, date: { type: Date, // `Date.now()` returns the current unix timestamp as a number default: Date.now } }); var BlogPost = db.model('BlogPost', schema); var post = new BlogPost({title: '5 Best Arnold Schwarzenegger Movies'}); // The post has a default Date set to now assert.ok(post.date.getTime() >= Date.now() - 1000); assert.ok(post.date.getTime() <= Date.now());
默认情况下,mongoose只在创建新document时应有默认值。如果你使用update()
和 findOneAndUpdate()它不会设置默认值。可是 mongoose 4.x 让你选择性加入这个行为使用 setDefaultsOnInsert 选项。
setDefaultsOnInsert 选项依赖MongoDB $setOnInsert 操作符。在 MongoDB 2.4中介绍了$setOnInsert操作符,如果你使用的MongoDB服务器<2.4.0,不要使用setDefaultsOnInsert。
var schema = new Schema({ title: String, genre: {type: String, default: 'Action'} }); var Movie = db.model('Movie', schema); var query = {}; var update = {title: 'The Terminator'}; var options = { // Return the document after updates are applied new: true, // Create a document if one isn't found. Required // for `setDefaultsOnInsert` upsert: true, setDefaultsOnInsert: true }; Movie. findOneAndUpdate(query, update, options, function (error, doc) { assert.ifError(error); assert.equal(doc.title, 'The Terminator'); assert.equal(doc.genre, 'Action'); });