前言
不知道大家知不知道,mongoose為數(shù)據(jù)模型提供了虛擬屬性, 借此可以更加一致地、方便地讀寫模型屬性,類似于C#或Java中的訪問器。 我們知道虛擬屬性在Query階段一定是查不到的,因為事實上MongoDB并沒有存儲這些屬性。 但是否可以通過一個攔截器來實現(xiàn)虛擬屬性的查詢呢?
這個問題很有趣,而且在很多場景下都相當方便。例如:
事實上,虛擬屬性查詢和虛擬屬性讀寫都是為了代碼復用。
Mongoose 中的 Hook
Mongoose Schema幾乎所有靜態(tài)方法和對象方法都添加了 .pre和.post鉤子。 這些鉤子其實就是函數(shù)鉤子,采用hooks-js的實現(xiàn)。
來自官網(wǎng)的例子:
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中實現(xiàn) ///////////////////////////////////////////////////////////////////// // 下面的代碼則是mongoose提供的Hook API // Define a middleware function to be invoked before 'save' Document.pre('save', function validate(next) { // ... });
在Document.save()
被調用時,上述validate函數(shù)就會被回調。
添加查詢鉤子
Mongoose沒有對hooks-js進一步封裝,這意味著我們不能對所有Query方法設置鉤子, 只能一一枚舉需要監(jiān)視的方法。當然,這不影響我們進行代碼復用。
// 設置 findOne 和 find 鉤子 CompanySchema.pre('findOne', preFind).pre('find', preFind);
接下來便著手實現(xiàn)preFind函數(shù)。
實現(xiàn)虛擬查詢
在鉤子(preFind)中,我們可以更改查詢條件借此實現(xiàn)虛擬查詢。 值得注意的是,完全可控的Query意味著我們可以實現(xiàn)任何形式的虛擬查詢。
例如全文檢索:
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 }] }); }
總結
聲明:本網(wǎng)頁內容旨在傳播知識,若有侵權等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com