webpack源碼中一些精妙的方法總結(jié)
前言
過年這一段時間一直在研究webpack的源碼,由于過年周圍氣氛比較歡快,心態(tài)有點飄導(dǎo)致沒有沉下心來仔細研究其中的細節(jié)。經(jīng)過反思之后,靜心重新捋順webpack的源碼,這時發(fā)現(xiàn)不少巧妙的方法值得學(xué)習(xí)。這里我已經(jīng)迫不及待的跟大家分享了,希望對大家平常開發(fā)過程中有所幫助。
精妙方法
緩存函數(shù)
這個方法最精妙的地方在于將執(zhí)行結(jié)果緩存,減少函數(shù)的重復(fù)執(zhí)行以達到提升性能的目的,對于執(zhí)行越復(fù)雜越耗時的函數(shù)收益越大。但是,不適用于動態(tài)執(zhí)行結(jié)果的函數(shù)。
const memoize = fn => { let cache = false; let result = undefined; return () => { if (cache) { return result; } else { result = fn(); cache = true; fn = undefined; return result; } }; };
這個方法跟惰性函數(shù)有點相似,只在函數(shù)第一次調(diào)用的時候執(zhí)行,將fn()的執(zhí)行結(jié)果緩存到result上,然后通過cache設(shè)置為true來標記緩存已開啟。這里還有個細節(jié)值得學(xué)習(xí):由于閉包的原因,fn方法被新的函數(shù)持有,一直在調(diào)用棧中得不到釋放,而在代碼中有一句fn = undefined,手動釋放內(nèi)存。
屬性劫持
這個方法通過自定義get方法或者value值來重定義obj的屬性。實現(xiàn)的結(jié)果有點類似于Object.freeze,但又不完全相同。通過Object.defineProperty定義name的get屬性描述符來保證obj的值永遠不變;通過Object.defineProperty來將name的writable設(shè)置為false來保證obj的value不會被改變。
const mergeExports = (obj, exports) => { ... for (const name of Object.keys(descriptors)) { const descriptor = descriptors[name]; if (descriptor.get) { const fn = descriptor.get; Object.defineProperty(obj, name, { configurable: false, enumerable: true, get: memoize(fn) }); } else if (typeof descriptor.value === "object") { Object.defineProperty(obj, name, { configurable: false, enumerable: true, writable: false, value: mergeExports({}, descriptor.value) }); } else { ... } } return Object.freeze(obj); };
這里還有個小的知識點:通過將屬性的configurable屬性描述符設(shè)置為false來保證屬性不可被刪除。
數(shù)組比較
進行兩個數(shù)組是否相等的比較。想必大家都知道,如果使用==進行數(shù)組比較的話,是進行引用地址的比較,所以想要判斷兩個數(shù)組是沒有方法進行直接比較的。而webpack源碼中的這個方法給我提供了一種數(shù)組比較的方式。當然了,這個方法只適用于扁平化的一維基本類型數(shù)組,如果想要比較復(fù)雜的情況的話,需要在for循環(huán)的基礎(chǔ)上稍加改造。
exports.equals = (a, b) => { if (a.length !== b.length) return false; for (let i = 0; i < a.length; i++) { if (a[i] !== b[i]) return false; } return true; };
配置項校驗
webpack中的實現(xiàn)的代碼量比較多,我們就只說一下大體實現(xiàn)思路,通過定義需要校驗的所有配置項的Schema校驗規(guī)則。大體Schema格式如下:
"Amd": { "description": "Set the value of `require.amd` and `define.amd`. Or disable AMD support.", "anyOf": [ { "description": "You can pass `false` to disable AMD support.", "enum": [false] }, { "description": "You can pass an object to set the value of `require.amd` and `define.amd`.", "type": "object" } ] },
其中key對應(yīng)的是需要校驗的配置項,value中的desciption對應(yīng)的是提示信息,其余部分對應(yīng)的是校驗規(guī)則。
結(jié)尾
到此這篇關(guān)于webpack源碼中一些精妙的方法的文章就介紹到這了,更多相關(guān)webpack源碼的精妙方法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Google 爬蟲如何抓取 JavaScript 的內(nèi)容
我們測試了谷歌爬蟲是如何抓取 JavaScript,下面就是我們從中學(xué)習(xí)到的知識,需要的朋友可以參考下2017-04-04關(guān)于better-scroll插件的無法滑動bug(2021通過插件解決)
這篇文章主要介紹了關(guān)于better-scroll插件的無法滑動bug(2021通過插件解決),本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-03-03