摘要:源碼分析四模塊上一篇我們看到,通過對命令行傳入的參數和配置文件里的配置項做了轉換包裝,然后傳遞給的模塊去編譯。這一篇我們來看看做了些什么事。在上面的分析中,我們看到最核心的其實就是實例,接下來我們就看下它的類的內部邏輯。
webpack 源碼分析(四)——complier模塊
上一篇我們看到,webpack-cli 通過 `yargs 對命令行傳入的參數和配置文件里的配置項做了轉換包裝,然后傳遞給 webpack 的 compiler 模塊去編譯。
這一篇我們來看看 compiler 做了些什么事。
入口首先,我們來看看 webpack 是在什么地方引入 Compiler 這個模塊的,在 webpack/lib/webpack.js 中我們發現,這個文件第一行就引入了它,并在下面的主要邏輯中使用了它:
const Compiler = require("./Compiler"); …… /* * options:這就是由 webpack-cli 轉換包裝后的配置項 * callback:第二篇分析時,發現 `webpack-cli` 只是傳入了 options對象來獲取 `compiler` 對象,并沒有傳入回調 */ const webpack = (options, callback) => { // 驗證配置項的格式 // webpackOptionsSchema 是一個json格式的描述文件,它描述了webpack可接受的所有配置項及其格式 // options 是用戶定義的webpack.config.*.js中導出的所有配置項 // validateSchema 使用 ajv 包,根據 webpackOptionsSchema 中定義的數據類型和描述來校驗 options 中的各項配置項,最后返回一個錯誤對象,其中包含所有錯誤的配置項及說明 const webpackOptionsValidationErrors = validateSchema( webpackOptionsSchema, options ); // 如果存在配置項錯誤,則拋出所有錯誤 if (webpackOptionsValidationErrors.length) { throw new WebpackOptionsValidationError(webpackOptionsValidationErrors); } //判斷配置項是否數組,如果是數組則使用MultiCompiler進行編譯,否則使用Compiler模塊進行編譯,一般情況下,我們的 options 都是對象不是數組 let compiler; if (Array.isArray(options)) { compiler = new MultiCompiler(options.map(options => webpack(options))); } else if (typeof options === "object") { //使用默認配置項處理輸入配置項 // WebpackOptionsDefaulter 繼承自 OptionsDefaulter 類 // 在這個類里有兩個對象: // * this.defaults : 用來存放配置項的值 // * this.config : 用來存放配置項的值的類型 // process 方法的處理邏輯: // * 如果defaults對象中存在,但在config對象中不存在,并且在options中也不存在,就從 defaults對象中復制一份到options中 // * 如果在config對象中存在并且值為 “call”,則說明它的值是一個方法調用,就直接調用,并將options作為參數傳入 // * 如果在config對象中存在且值為 “make”,并且在 options中沒有,則說明它是一個方法,就直接調用它,并將options作為參數傳入,拿到返回值,賦值給options中對應的項 // * 如果在config對象中存在切值為 “append”,則取出options中對應的值,如果它不是數組,就把它重置為數組,并且把defaults對象中的值復制到數組中,最后將這個數組作為值賦值給options相應的項 options = new WebpackOptionsDefaulter().process(options); // new 一個compiler實例,參數為當前執行node命令的目錄路徑 // Compiler類繼承自我們上一篇講過的Tapable類,在構造函數中,初始化了各種類型的鉤子實例 // compiler 類的內部邏輯,后面詳解++++++++++++++++++++++ compiler = new Compiler(options.context); // WebpackOptionsDefaulter類處理后返回的options 復制給 compiler compiler.options = options; // 使用NodeEnviromentPlugin 類給compiler添加文件輸入輸出的能力 // NodeEnviromentPlugin的內部邏輯,后面詳解++++++++++++++++++++++++++ new NodeEnvironmentPlugin().apply(compiler); // 如果配置項中有插件配置并且插件配置為數組 // 遍歷插件數組,如果插件是一個函數,則使用complier來調用它,,并且將compier作為參數出入 // 否則,使用 WebpackPluginInstance 的 apply 方法來返回一個 void 值(相當于undefined) if (options.plugins && Array.isArray(options.plugins)) { for (const plugin of options.plugins) { if (typeof plugin === "function") { plugin.call(compiler, compiler); } else { plugin.apply(compiler); } } } // 觸發 environment 同步鉤子,這里的 call() 是我們講過的Tapable鉤子事件的觸發方法 // environment 準備好之后,執行插件 compiler.hooks.environment.call(); // 觸發 afterEnvironment 同步鉤子 // environment 安裝完成之后,執行插件 compiler.hooks.afterEnvironment.call(); // 使用 WebpackOptionsApply 類處理選項,返回處理過的選項對象 // WebpackOptionsApply 的處理邏輯,后面詳解++++++++++++++ compiler.options = new WebpackOptionsApply().process(options, compiler); } else { //如果配置既不是數組類型也不是對象類型,拋出錯誤 throw new Error("Invalid argument: options"); } // 如果傳入了回到函數 if (callback) { // 如果回調不是函數,拋出參數類型錯誤 if (typeof callback !== "function") { throw new Error("Invalid argument: callback"); } // 如果是監聽模式,或者(配置是數組且數組中的項中有監聽選項為true),則初始化監聽配置,最后返回compiler實例的監聽方法 // 實例的監聽方法會返回一個 Watching類的實例,它本質上是一個觀察者 if ( options.watch === true || (Array.isArray(options) && options.some(o => o.watch)) ) { const watchOptions = Array.isArray(options) ? options.map(o => o.watchOptions || {}) : options.watchOptions || {}; return compiler.watch(watchOptions, callback); } // 使用compiler的run方法運行回調 compiler.run(callback); } //返回compiler實例 return compiler; };
通過以上源碼分析,我們得知,在 webpack() 中,實際上總共做了三件事:
對參數進行校驗和規范化處理;
new 一個編譯器實例并且初始化各種tapable鉤子,并且在環境準備好和安裝完成后執行響應的鉤子
初始化監聽。
在上面的分析中,我們看到最核心的其實就是compiler實例,接下來我們就看下它的類的內部邏輯。
compiler 分析 主體結構首先,我們來看主要結構:
/*…… *// 各種引入 // Compiler 類繼承自Tapable class Compiler extends Tapable { constructor(context) {……}//構造函數執行各種初始化操作 watch(watchOptions, handler) {……}//監聽初始化 run(callback) {……}// 運行編譯 runAsChild(callback) {...} // 作為子編譯進程運行 purgeInputFileSystem() {...} // 凈化輸入 emitAssets(compilation, callback) {...} // 發布資源 emitRecords(callback) {...} // 發布記錄 readRecords(callback) {...} // 讀取記錄 createChildCompiler( compilation, compilerName, compilerIndex, outputOptions, plugins ) {...} // 創建子編譯器 isChild() {return !!this.parentCompilation;} // 是否子匯編 createCompilation() {return new Compilation(this);} // 創建匯編實例 newCompilation(params) {return compilation;} // 根據參數創建新的匯編實例 createNormalModuleFactory() {return normalModuleFactory;} // 創建普通模塊的工廠 createContextModuleFactory() {return contextModuleFactory;} // 創建上下文模塊的工廠 newCompilationParams() {return params;} // 獲取一個新的匯編參數對象 compile(callback) {} // 編譯 } module.exports = Compiler; class SizeOnlySource extends Source {} // 定義了一個類,作用是+++++++++++
下一講,我們將逐個攻破
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/110319.html
摘要:基本配置項基本配置項。的插件架構主要基于實現的,這個就是專注于事件的廣播和操作。開啟多進程,加快打包速度。 這次我們主要研究的是webpack框架的相關知識,webpack是一個打包構建的前端框架,用于解決前端開發的模塊化問題。 應用場景和縱向比較 說到webpack,肯定你還會想到gulp和grunt這些框架,那么webpack是做什么的呢?他和其他的框架有什么區別呢?我們一起來分析...
摘要:不直接使用的原因很簡單首先構建一次實在太慢了,特別是有幾十個頁面存在的情況下,另一個原因是我只是想拿到資源依賴,我根本不想對整個前端進行一次構建,也不想生成任何。這就達到了本文題目中目的,用十分之一的構建時間做一場頁面靜態資源依賴分析。原文鏈接 作者:梯田 前言: 所謂【靜態資源依賴分析】,指的是可以通過分析頁面資源后,可以以 json 數據或者圖表的方式拿到頁面資源間的依賴關系。 比如 c...
摘要:類定義了方法,用于注冊插件,將插件及其回調函數以的形式保存在內部對象中又定義了,等方法來觸發插件的回調函數。所以當類繼承類后,也同樣具有注冊插件和觸發回調函數的功能。 說起webpack,相信對于前端工程師們而言早已經不是什么新鮮的事物。但是由于webpack有著較為復雜和靈活的配置項,所以給人的第一感覺是難以完全掌握。 這次就跟大家分享一下有關webpack構建過程的相關知識,希望對...
摘要:以為例,編寫來幫助我們完成重復的工作編譯壓縮我只要執行一下就可以檢測到文件的變化,然后為你執行一系列的自動化操作,同樣的操作也發生在這些的預處理器上。的使用是針對第三方類庫使用各種模塊化寫法以及語法。 showImg(https://segmentfault.com/img/bVbtZYK); 一:前端工程化的發展 很久以前,互聯網行業有個職位叫做 軟件開發工程師 在那個時代,大家可能...
摘要:需要得到最后一個產生的處理結果。這個處理結果應該是或者被轉換為一個,代表了模塊的源碼。另外還可以傳遞一個可選的結果格式為對象。在異步模式中,必須調用,來指示等待異步結果,它會返回回調函數,隨后必須返回并且調用該回調函數。 準備工作 安裝 Node.js, 建議安裝LTS長期支持版本 mkdir webpack and cd webpack and npm init -y npm ...
閱讀 804·2023-04-25 19:40
閱讀 3405·2023-04-25 17:41
閱讀 2993·2021-11-11 11:01
閱讀 2587·2019-08-30 15:55
閱讀 3217·2019-08-30 15:44
閱讀 1347·2019-08-29 14:07
閱讀 478·2019-08-29 11:23
閱讀 1314·2019-08-27 10:54