Mongoose 是 Node.js 的一个 ORM 框架,用于简化与 MongoDB 数据库的交互。Mongoose 提供了许多有用的功能,其中包含生命周期 hooks。在本文中,我们将介绍这些生命周期 hooks 并详细阐述它们在 Mongoose 中的使用方法。
什么是生命周期 hooks?
生命周期 hooks 是在 Mongoose 模型的操作期间调用的函数。这些函数允许开发人员对模型的不同操作进行重写、添加或拦截。Mongoose 生命周期 hooks 分为四种类型:
- Save Hooks:在保存文档之前或之后运行,这些 hooks 可以用于验证和修改文档。
- Query Hooks:在执行查询之前或之后运行,这些 hooks 可以用于修改查询参数或添加附加操作。
- Aggregate Hooks:在聚合操作之前或之后运行,这些 hooks 可以用于自定义聚合操作以及聚合结果。
- Document Hooks:在文档初始化、验证、保存、删除之前或之后运行,这些 hooks 可以用于添加自定义逻辑到文档操作中。
如何使用生命周期 hooks
在 Mongoose 中,可以通过在模型的 Schema 中定义 hooks 来使用生命周期 hooks。下面是一个简单的示例:
-- -------------------- ---- ------- ----- ---------- - --- ----------------- ----- ------- ------ ------- --- ---------------------- -------- ------ - ------------------- ---------- ------- --- ----- ---- - ---------------------- ------------ ----- ---- - --- ------ ----- ----- ----- ------ ---------------- --- ------------展开代码
在这个示例中,我们定义了一个 User
模型,并通过 pre
方法向其添加了一个 save
hook。在这个 hook 中,我们打印出一个日志信息并调用了 next
函数以继续执行操作。
当我们执行 user.save()
操作时,我们会在控制台中看到以下输出:
Saving user...
生命周期 hooks 的详细说明
Save Hooks
Save Hooks 是一种用于在保存文档之前或之后运行的 Mongoose 生命周期 hooks。Save Hooks 可以在模型上执行两个操作:pre('save')
和 post('save')
。
pre('save')
pre('save')
hook 在保存文档之前执行。这个 hook 可以用于对文档进行验证或修改。如果 next(error)
被调用了,Mongoose 将在保存文档时抛出一个错误。
下面是一个示例,说明如何在保存文档之前对文档进行验证:
-- -------------------- ---- ------- ---------------------- -------- ------ - -- ----------------------------- - -- ---------- -- ---------- ------------------ -------- ----- ----- - -- ----- - ------ ---------- - -------------------------- ----- -------- ----- ----- - -- ----- - ------ ---------- - ------------- - ----- -- --------- ------- --- --- - ---- - ------- - ---展开代码
这个 hook 会在保存文档之前检查用户是否正在修改密码。如果是,则使用 bcrypt 将密码哈希化,并将哈希值保存为密码。
post('save')
post('save')
hook 在保存文档之后执行。这个 hook 可以用于在保存后处理文档,也可以跟踪用户对文档的操作。
下面是一个示例,说明如何在保存文档之后跟踪用户对文档的操作:
userSchema.post('save', function (doc, next) { if (doc.isNew) { // 如果文档是新创建的 console.log('New user created'); } else { console.log('User updated'); } next(); });
这个 hook 会在保存文档之后打印出消息,以表示用户对文档的操作状态。如果文档是新创建的,就会打印出 "New user created",否则就会打印出 "User updated"。
Query Hooks
Query Hooks 是一种用于在执行查询之前或之后运行的 Mongoose 生命周期 hooks。Query Hooks 可以在模型上执行四个操作:pre('find')
、pre('findOne')
、pre('update')
和 pre('remove')
。
pre('find') 和 pre('findOne')
pre('find')
和 pre('findOne')
hooks 在执行查询之前执行,并且可以用于修改查询参数。
下面是一个示例,说明如何在查询参数中引入一个附加过滤器:
userSchema.pre('find', function () { this.where({ isDeleted: { $ne: true } }); });
这个 hook 会在查询之前将 { isDeleted: { $ne: true } }
添加到查询条件中,从而排除已删除的用户。
pre('update')
pre('update')
hook 在更新文档之前执行,并且可以用于修改更新操作的查询参数。
下面是一个示例,说明如何在更新文档之前将修改者信息添加到更新集合中:
userSchema.pre('update', function () { this.set({ updatedAt: new Date() }); // 更新修改时间 if (this._update.$set) { // 如果 $set 操作被使用 this.set({ 'updatedBy.id': this.context.user.id }); this.set({ 'updatedBy.name': this.context.user.name }); } });
这个 hook 在更新集合之前将 updatedAt
操作和 updatedBy
操作添加到集合中。如果使用了 $set
操作符,则会添加到 updatedBy
操作和 updatedAt
操作中。
pre('remove')
pre('remove')
hook 在删除文档之前执行,并且可以用于在删除操作之前进行验证或拦截。
下面是一个示例,说明如何在删除文档之前进行验证:
-- -------------------- ---- ------- ------------------------ -------- ------ - -- ------------------ ------------- ------- --------- ------- - ---- ----------- - -- -------- ----- ------ - -- ----- - ------ ---------- - -- ------ - -- - ------ -------- ------------- ------ ---- ---- ------- ---------- - ---- - ------- - --- ---展开代码
这个 hook 在删除文档之前会检查用户是否有未完成的订单。如果有,则会抛出一个错误并阻止删除操作。如果没有,则可以继续进行删除操作。
Aggregate Hooks
Aggregate Hooks 是一种用于在执行聚合操作之前或之后运行的 Mongoose 生命周期 hooks。Aggregate Hooks 可以在模型上执行两个操作:pre('aggregate')
和 post('aggregate')
。
pre('aggregate')
pre('aggregate')
hook 在执行聚合操作之前执行,并且可以用于添加自定义聚合操作。
下面是一个示例,说明如何向聚合操作中添加附加操作:
userSchema.pre('aggregate', function () { this.append({ $match: { isActive: true } }); // 添加 $match 操作符 });
这个 hook 在聚合操作之前添加了一个 $match
操作符,从而只返回已激活的用户。
post('aggregate')
post('aggregate')
hook 在执行聚合操作之后执行,并且可以用于处理聚合结果。
下面是一个示例,说明如何在聚合操作之后处理结果:
userSchema.post('aggregate', function (result) { console.log(result); // 打印聚合结果 });
这个 hook 在聚合操作之后输出聚合结果。
Document Hooks
Document Hooks 是一种用于在文档初始化、验证、保存、删除之前或之后运行的 Mongoose 生命周期 hooks。Document Hooks 可以在模型上执行八个操作:pre('init')
、post('init')
、pre('validate')
、post('validate')
、pre('save')
、post('save')
、pre('remove')
和 post('remove')
。
pre('init') 和 post('init')
pre('init')
和 post('init')
hooks 在初始化文档之前和之后执行,可以用于对文档进行设置。在这些 hooks 中,无法更改模型的验证规则。
下面是一个示例,说明如何在初始化文档之前添加自定义属性到文档中:
userSchema.pre('init', function () { this.fullName = `${this.firstName} ${this.lastName}`; }); userSchema.post('init', function (doc) { console.log(`New user initialized: ${doc.fullName}`); });
这个 hook 添加了一个 fullName
属性到文档中,并在初始化之后打印出消息。
pre('validate') 和 post('validate')
pre('validate')
和 post('validate')
hooks 在完成对文档的非自定义验证之前和之后执行。
下面是一个示例,说明如何在验证文档之前和之后添加自定义验证:
-- -------------------- ---- ------- -------------------------- -------- -- - -- ------------ - ---------- - ------------------------- - --- --------------------------- -------- ----- - -- -------- - --- - ------- - --- - ---展开代码
这个 hook 在验证文档之前将电子邮件地址转换为小写,并在验证之后将未满 18 岁的用户的年龄强制设为 18 岁。
pre('save') 和 post('save')
pre('save')
和 post('save')
hooks 在保存文档之前和之后执行,并可以用于在文档被保存之前和之后进行后期处理。
下面是一个示例,说明如何在保存文档之前和之后添加自定义逻辑:
-- -------------------- ---- ------- ---------------------- -------- ------ - -- --------------- - ------------ - --- ------- -- ----------------------- - ------- --- ----------------------- -------- ----- - ----------------- ------ ------------- ---展开代码
这个 hook 在保存文档之前检查文档是否已经创建,并在保存之后打印出消息。
pre('remove') 和 post('remove')
pre('remove')
和 post('remove')
hooks 在删除文档之前和之后执行,并可以用于在文档被删除之前和之后进行后期处理。
下面是一个示例,说明如何在删除文档之前和之后添加自定义逻辑:
-- -------------------- ---- ------- ------------------------ -------- ------ - ------------------ ------- -------- -- -------- ----- - -- ------------ -- ----- - ------ ---------- - ------- --- --- ------------------------- -------- ----- - ----------------- -------- ------------- ---展开代码
这个 hook 在删除文档之前删除与当前用户关联的订单,并在删除之后打印出消息。
结论
在 Mongoose 中使用生命周期 hooks 使开发人员能够在文档的操作期间重写、添加或拦截逻辑。在本文中,我们介绍了四种类型的 hooks,包含它们在 Mongoose 中的使用方法以及示例代码。使用生命周期 hooks,开发人员可以将定制处理逻辑集成到 Mongoose 模型中,从而提高代码的可重用性和可维护性。
来源:JavaScript中文网 ,转载请注明来源 https://www.javascriptcn.com/post/67063f51d91dce0dc85a8811