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浏览器扫码功能使用地址