Mongoose实现虚拟字段查询的方法详解
发布时间 - 2026-01-11 02:47:44 点击率:次前言

不知道大家知不知道,mongoose为数据模型提供了虚拟属性, 借此可以更加一致地、方便地读写模型属性,类似于C#或Java中的访问器。 我们知道虚拟属性在Query阶段一定是查不到的,因为事实上MongoDB并没有存储这些属性。 但是否可以通过一个拦截器来实现虚拟属性的查询呢?
这个问题很有趣,而且在很多场景下都相当方便。例如:
- 实现一个暴力的全文检索时,需要对多个字段匹配统一查询词,该查询词可抽象为虚拟属性;
- 多处都需要进行同一个复杂条件的查询时,可以用虚拟属性封装该查询条件。
事实上,虚拟属性查询和虚拟属性读写都是为了代码复用。
Mongoose 中的 Hook
Mongoose Schema几乎所有静态方法和对象方法都添加了 .pre和.post钩子。 这些钩子其实就是函数钩子,采用hooks-js的实现。
来自官网的例子:
var hooks = require('hooks')
, Document = require('./path/to/some/document/constructor');
// Add hooks' methods: `hook`, `pre`, and `post`
for (var k in hooks) {
Document[k] = hooks[k];
}
// Define a new method that is able to invoke pre and post middleware
Document.hook('save', Document.prototype.save);
// 上述代码在mongoose中实现
/////////////////////////////////////////////////////////////////////
// 下面的代码则是mongoose提供的Hook API
// Define a middleware function to be invoked before 'save'
Document.pre('save', function validate(next) {
// ...
});
在Document.save()被调用时,上述validate函数就会被回调。
添加查询钩子
Mongoose没有对hooks-js进一步封装,这意味着我们不能对所有Query方法设置钩子, 只能一一枚举需要监视的方法。当然,这不影响我们进行代码复用。
// 设置 findOne 和 find 钩子
CompanySchema.pre('findOne', preFind).pre('find', preFind);
接下来便着手实现preFind函数。
实现虚拟查询
在钩子(preFind)中,我们可以更改查询条件借此实现虚拟查询。 值得注意的是,完全可控的Query意味着我们可以实现任何形式的虚拟查询。
例如全文检索:
function preFind() {
var word = this.getQuery().word;
if(word === undefined) return;
// 从真实的Query中删掉虚拟属性
delete this._conditions.word;
// 构造正则表达式
var regex = new RegExp(word);
// 全文检索
this.where({ $or: [{ title: regex }, { content: regex }, { author: regex }] });
}
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。
# mongoose
# 字段
# 添加字段
# mongoose查询指定字段
# Node.js中使用mongoose操作mongodb数据库的方法
# Mongoose学习全面理解(推荐)
# 详解Nodejs基于mongoose模块的增删改查的操作
# 利用Mongoose让JSON数据直接插入或更新到MongoDB
# Mongodb 数据类型及Mongoose常用CURD
# MongoDB用Mongoose得到的对象不能增加属性完美解决方法(两种)
# Node.js的MongoDB驱动Mongoose基本使用教程
# 详解Nodejs mongoose
# Vue+Node实现商品列表的分页、排序、筛选
# 添加购物车功能详解
# 用vue和node写的简易购物车实现
# node.js使用mongoose操作数据库实现购物车的增、删、改、查功能示例
# 复用
# 的是
# 都是
# 事实上
# 就会
# 多个
# 则是
# 可以用
# 这个问题
# 我们可以
# 可以通过
# 可以实现
# 这篇文章
# 谢谢大家
# 几乎所有
# 来实现
# 类似于
# 能对
# 回调
# 多处
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
郑州企业网站制作公司,郑州招聘网站有哪些?
jQuery validate插件功能与用法详解
js代码实现下拉菜单【推荐】
开心动漫网站制作软件下载,十分开心动画为何停播?
手机怎么制作网站教程步骤,手机怎么做自己的网页链接?
Laravel如何使用Laravel Vite编译前端_Laravel10以上版本前端静态资源管理【教程】
Laravel如何发送系统通知_Laravel Notifications实现多渠道消息通知
js实现获取鼠标当前的位置
零服务器AI建站解决方案:快速部署与云端平台低成本实践
消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工
Laravel如何与Docker(Sail)协同开发?(环境搭建教程)
黑客如何利用漏洞与弱口令入侵网站服务器?
Laravel怎么配置不同环境的数据库_Laravel本地测试与生产环境动态切换【方法】
Laravel怎么实现搜索功能_Laravel使用Eloquent实现模糊查询与多条件搜索【实例】
Laravel如何使用Blade组件和插槽?(Component代码示例)
极客网站有哪些,DoNews、36氪、爱范儿、虎嗅、雷锋网、极客公园这些互联网媒体网站有什么差异?
Laravel如何连接多个数据库_Laravel多数据库连接配置与切换教程
英语简历制作免费网站推荐,如何将简历翻译成英文?
如何在阿里云购买域名并搭建网站?
HTML透明颜色代码怎么让图片透明_给img元素加透明色的技巧【方法】
音乐网站服务器如何优化API响应速度?
Laravel用户认证怎么做_Laravel Breeze脚手架快速实现登录注册功能
阿里云网站搭建费用解析:服务器价格与建站成本优化指南
在centOS 7安装mysql 5.7的详细教程
奇安信“盘古石”团队突破 iOS 26.1 提权
,网页ppt怎么弄成自己的ppt?
如何在HTML表单中获取用户输入并用JavaScript动态控制复利计算循环
如何挑选最适合建站的高性能VPS主机?
Linux网络带宽限制_tc配置实践解析【教程】
南京网站制作费用,南京远驱官方网站?
Laravel如何实现API速率限制?(Rate Limiting教程)
googleplay官方入口在哪里_Google Play官方商店快速入口指南
网站建设保证美观性,需要考虑的几点问题!
Laravel Seeder怎么填充数据_Laravel数据库填充器的使用方法与技巧
Laravel如何实现邮件验证激活账户_Laravel内置MustVerifyEmail接口配置【步骤】
Laravel Pest测试框架怎么用_从PHPUnit转向Pest的Laravel测试教程
php静态变量怎么调试_php静态变量作用域调试技巧【解答】
微信小程序 input输入框控件详解及实例(多种示例)
香港服务器如何优化才能显著提升网站加载速度?
怎么制作一个起泡网,水泡粪全漏粪育肥舍冬季氨气超过25ppm,可以有哪些措施降低舍内氨气水平?
如何在万网利用已有域名快速建站?
如何用wdcp快速搭建高效网站?
Android自定义控件实现温度旋转按钮效果
javascript基本数据类型及类型检测常用方法小结
Laravel如何实现邮箱地址验证功能_Laravel邮件验证流程与配置
Laravel中间件起什么作用_Laravel Middleware请求生命周期与自定义详解
谷歌Google入口永久地址_Google搜索引擎官网首页永久入口
昵图网官网入口 昵图网素材平台官方入口
晋江文学城电脑版官网 晋江文学城网页版直接进入
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址

