怎么在JavaScript中實現(xiàn)鏈?zhǔn)秸{(diào)用?相信很多沒有經(jīng)驗的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。
十載的尉犁網(wǎng)站建設(shè)經(jīng)驗,針對設(shè)計、前端、開發(fā)、售后、文案、推廣等六對一服務(wù),響應(yīng)快,48小時及時工作處理。網(wǎng)絡(luò)營銷推廣的優(yōu)勢是能夠根據(jù)用戶設(shè)備顯示端的尺寸不同,自動調(diào)整尉犁建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設(shè)計,從而大程度地提升瀏覽體驗。成都創(chuàng)新互聯(lián)從事“尉犁網(wǎng)站設(shè)計”,“尉犁網(wǎng)站推廣”以來,每個客戶項目都認(rèn)真落實執(zhí)行。
什么是鏈?zhǔn)秸{(diào)用
鏈?zhǔn)秸{(diào)用在 JavaScript 語言界很常見,如 jQuery 、 Promise 等,都是使用的鏈?zhǔn)秸{(diào)用。鏈?zhǔn)秸{(diào)用可以讓我們在進(jìn)行連續(xù)操作時,寫出更簡潔的代碼。
new Promise((resolve, reject) => { resolve(); }) .then(() => { throw new Error('Something failed'); }) .then(() => { console.log('Do this whatever happened before'); }) .catch(() => { console.log('Do that'); })
逐步實現(xiàn)鏈?zhǔn)秸{(diào)用
假設(shè),我們要實現(xiàn)一個 math 模塊,使之能夠支持鏈?zhǔn)秸{(diào)用:
const math = require('math'); const a = math.add(2, 4).minus(3).times(2); const b = math.add(2, 4).times(3).divide(2); const c = { a, b }; console.log(a.times(2) + b + 1); // 22 console.log(a.times(2) + b + 2); // 23 console.log(JSON.stringify(c)); // {"a":6,"b":9}
基本的鏈?zhǔn)秸{(diào)用
鏈?zhǔn)秸{(diào)用通常的實現(xiàn)方式,就是在函數(shù)調(diào)用結(jié)果返回模塊本身。那么 math 模塊的代碼大致應(yīng)該是這樣子的:
export default { add(...args) { // add return this; }, minus(...args) { // minus return this; }, times(...args) { // times return this; }, divide(...args) { // divide return this; }, }
方法如何返回值
上述代碼實現(xiàn)了鏈?zhǔn)秸{(diào)用,但是也存在一個問題,就是無法獲取計算結(jié)果。所以我們需要對模塊進(jìn)行改造,使用一個內(nèi)部變量來存儲計算結(jié)果。
export default { value: NaN, add(...args) { this.value = args.reduce((pv, cv) => pv + cv, this.value || 0); return this; }, }
這樣,我們在最后一步,通過 .value 就可以拿到最終的計算結(jié)果了。
問題真的解決了嗎
上面我們看似通過一個 value 變量解決了存儲計算結(jié)果的問題,但是發(fā)生第二次鏈?zhǔn)秸{(diào)用時, value 的值因為已經(jīng)有了初始值,我們會得到錯誤的計算結(jié)果!
const a = math.add(5, 6).value; // 11 const b = math.add(5, 7).value; // 23 而非 12
既然是因為 value 有了初始值,那么能不能在獲取 value 的值時重置掉呢?答案是不能,因為我們并不能確定使用者會在什么時候取值。
另外一種思路是在每次鏈?zhǔn)秸{(diào)用之前生成一個新的實例,這樣就可以確保實例之間相互獨(dú)立了。
const math = function() { if (!(this instanceof math)) return new math(); }; math.prototype.value = NaN; math.prototype.add = function(...args) { this.value = args.reduce((pv, cv) => pv + cv, this.value || 0); return this; }; const a = math().add(5, 6).value; const b = math().add(5, 7).value;
但是這樣也不能徹底解決問題,假設(shè)我們?nèi)缦抡{(diào)用:
const m = math().add(5, 6); const c = m.add(5).value; // 16 const d = m.add(5).value; // 21 而非 16
所以,最終要解決這個問題,只能每個方法都返回一個新的實例,這樣可確保無論怎么調(diào)用,相互之間都不會被干擾到。
math.prototype.add = function(...args) { const instance = math(); instance.value = args.reduce((pv, cv) => pv + cv, this.value || 0); return instance; };
如何支持不通過 .value 對結(jié)果進(jìn)行普通運(yùn)算
通過改造 valueOf 方法或者 Symbol.toPrimitive 方法。其中 Symbol.toPrimitive 方法優(yōu)先 valueOf 方法被調(diào)用,除非是ES環(huán)境不支持。
如何支持 JSON.stringify 序列化計算結(jié)果
通過自定義 toJSON 方法。 JSON.stringify 將值轉(zhuǎn)換為相應(yīng)的JSON格式時,如果被轉(zhuǎn)換值有 toJSON 方法,則優(yōu)先使用該方法返回的值。
最終的完整實現(xiàn)代碼
class Math { constructor(value) { let hasInitValue = true; if (value === undefined) { value = NaN; hasInitValue = false; } Object.defineProperties(this, { value: { enumerable: true, value: value, }, hasInitValue: { enumerable: false, value: hasInitValue, }, }); } add(...args) { const init = this.hasInitValue ? this.value : args.shift(); const value = args.reduce((pv, cv) => pv + cv, init); return new Math(value); } minus(...args) { const init = this.hasInitValue ? this.value : args.shift(); const value = args.reduce((pv, cv) => pv - cv, init); return new Math(value); } times(...args) { const init = this.hasInitValue ? this.value : args.shift(); const value = args.reduce((pv, cv) => pv * cv, init); return new Math(value); } divide(...args) { const init = this.hasInitValue ? this.value : args.shift(); const value = args.reduce((pv, cv) => pv / cv, init); return new Math(value); } toJSON() { return this.valueOf(); } toString() { return String(this.valueOf()); } valueOf() { return this.value; } [Symbol.toPrimitive](hint) { const value = this.value; if (hint === 'string') { return String(value); } else { return value; } } } export default new Math();
看完上述內(nèi)容,你們掌握怎么在JavaScript中實現(xiàn)鏈?zhǔn)秸{(diào)用的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!