細說 webpack 之流程篇

0.1. 引言

目前,幾乎全部業務的開發構建都會用到 webpack 。的確,做爲模塊加載和打包神器,只需配置幾個文件,加載各類 loader 就能夠享受無痛流程化開發。但對於 webpack 這樣一個複雜度較高的插件集合,它的總體流程及思想對咱們來講仍是很透明的。那麼接下來我會帶你瞭解 webpack 這樣一個構建黑盒,首先來談談它的流程。css

0.2. 準備工做

1. webstorm 中配置 webpack-webstorm-debugger-script

在開始瞭解以前,必需要能對 webpack 整個流程進行 debug ,配置過程比較簡單。node

先將 webpack-webstorm-debugger-script 中的 webstorm-debugger.js 置於 webpack.config.js 的同一目錄下,搭建好你的腳手架後就能夠直接 Debug 這個 webstorm-debugger.js 文件了。react

2. webpack.config.js 配置

估計你們對 webpack.config.js 的配置也嘗試過很多次了,這裏就大體對這個配置文件進行個分析。webpack

var path = require('path');
var node_modules = path.resolve(__dirname, 'node_modules');
var pathToReact = path.resolve(node_modules, 'react/dist/react.min.js');

module.exports = {
  // 入口文件,是模塊構建的起點,同時每個入口文件對應最後生成的一個 chunk。
  entry: {
    bundle: [
      'webpack/hot/dev-server',
      'webpack-dev-server/client?http://localhost:8080',
      path.resolve(__dirname, 'app/app.js')
    ]
  },
  // 文件路徑指向(可加快打包過程)。
  resolve: {
    alias: {
      'react': pathToReact
    }
  },
  // 生成文件,是模塊構建的終點,包括輸出文件與輸出路徑。
  output: {
    path: path.resolve(__dirname, 'build'),
    filename: '[name].js'
  },
  // 這裏配置了處理各模塊的 loader ,包括 css 預處理 loader ,es6 編譯 loader,圖片處理 loader。
  module: {
    loaders: [
      {
        test: /\.js$/,
        loader: 'babel',
        query: {
          presets: ['es2015', 'react']
        }
      }
    ],
    noParse: [pathToReact]
  },
  // webpack 各插件對象,在 webpack 的事件流中執行對應的方法。
  plugins: [
    new webpack.HotModuleReplacementPlugin()
  ]
};

 

除此以外再大體介紹下 webpack 的一些核心概念:git

  • loader:能轉換各種資源,並處理成對應模塊的加載器。loader 間能夠串行使用。
  • chunk:code splitting 後的產物,也就是按需加載的分塊,裝載了不一樣的 module。

對於 module 和 chunk 的關係能夠參照 webpack 官方的這張圖:es6

img

  • plugin:webpack 的插件實體,這裏以 UglifyJsPlugin 爲例。github

    function UglifyJsPlugin(options) {
      this.options = options;
    }
    
    module.exports = UglifyJsPlugin;
    
    UglifyJsPlugin.prototype.apply = function(compiler) {
      compiler.plugin("compilation", function(compilation) {
        compilation.plugin("build-module", function(module) {
        });
        compilation.plugin("optimize-chunk-assets", function(chunks, callback) {
          // Uglify 邏輯
        });
        compilation.plugin("normal-module-loader", function(context) {
        });
      });
    };

     

    在 webpack 中你常常能夠看到 compilation.plugin(‘xxx’, callback) ,你能夠把它看成是一個事件的綁定,這些事件在打包時由 webpack 來觸發。web

3. 流程總覽

在具體流程學習前,能夠先經過這幅 webpack 總體流程圖 瞭解一下大體流程(建議保存下來查看)。shell

img

0.3. shell 與 config 解析

每次在命令行輸入 webpack 後,操做系統都會去調用 ./node_modules/.bin/webpack 這個 shell 腳本。這個腳本會去調用 ./node_modules/webpack/bin/webpack.js 並追加輸入的參數,如 -p , -w 。(圖中 webpack.js 是 webpack 的啓動文件,而 $@ 是後綴參數)npm

img

在 webpack.js 這個文件中 webpack 經過 optimist 將用戶配置的 webpack.config.js 和 shell 腳本傳過來的參數整合成 options 對象傳到了下一個流程的控制對象中。

1. optimist

和 commander 同樣,optimist 實現了 node 命令行的解析,其 API 調用很是方便。

var optimist = require("optimist");

optimist
  .boolean("json").alias("json", "j").describe("json")
  .boolean("colors").alias("colors", "c").describe("colors")
  .boolean("watch").alias("watch", "w").describe("watch")
  ...

 

獲取到後綴參數後,optimist 分析參數並以鍵值對的形式把參數對象保存在 optimist.argv 中,來看看 argv 究竟有什麼?

// webpack --hot -w
{
  hot: true,
  profile: false,
  watch: true,
  ...
}

 

2. config 合併與插件加載

在加載插件以前,webpack 將 webpack.config.js 中的各個配置項拷貝到 options 對象中,並加載用戶配置在 webpack.config.js 的 plugins 。接着 optimist.argv 會被傳入到 ./node_modules/webpack/bin/convert-argv.js 中,經過判斷 argv 中參數的值決定是否去加載對應插件。(至於 webpack 插件運行機制,在以後的運行機制篇會提到)

ifBooleanArg("hot", function() {
  ensureArray(options, "plugins");
  var HotModuleReplacementPlugin = require("../lib/HotModuleReplacementPlugin");
  options.plugins.push(new HotModuleReplacementPlugin());
});
...
return options;

 

options 做爲最後返回結果,包含了以後構建階段所需的重要信息。

{ 
  entry: {},//入口配置
  output: {}, //輸出配置
  plugins: [], //插件集合(配置文件 + shell指令) 
  module: { loaders: [ [Object] ] }, //模塊配置
  context: //工程路徑
  ... 
}

 

這和 webpack.config.js 的配置很是類似,只是多了一些經 shell 傳入的插件對象。插件對象一初始化完畢, options 也就傳入到了下個流程中。

var webpack = require("../lib/webpack.js");
var compiler = webpack(options);

 

0.4. 編譯與構建流程

在加載配置文件和 shell 後綴參數申明的插件,並傳入構建信息 options 對象後,開始整個 webpack 打包最漫長的一步。而這個時候,真正的 webpack 對象纔剛被初始化,具體的初始化邏輯在 lib/webpack.js中,以下:

function webpack(options) {
  var compiler = new Compiler();
  ...// 檢查options,若watch字段爲true,則開啓watch線程
  return compiler;
}
...

 

webpack 的實際入口是 Compiler 中的 run 方法,run 一旦執行後,就開始了編譯和構建流程 ,其中有幾個比較關鍵的 webpack 事件節點。

  • compile 開始編譯
  • make 從入口點分析模塊及其依賴的模塊,建立這些模塊對象
  • build-module 構建模塊
  • after-compile 完成構建
  • seal 封裝構建結果
  • emit 把各個chunk輸出到結果文件
  • after-emit 完成輸出
1. 核心對象 Compilation

compiler.run 後首先會觸發 compile ,這一步會構建出 Compilation 對象:

compilation類圖

這個對象有兩個做用,一是負責組織整個打包過程,包含了每一個構建環節及輸出環節所對應的方法,能夠從圖中看到比較關鍵的步驟,如 addEntry() , _addModuleChain() ,buildModule() , seal() , createChunkAssets() (在每個節點都會觸發 webpack 事件去調用各插件)。二是該對象內部存放着全部 module ,chunk,生成的 asset 以及用來生成最後打包文件的 template 的信息。

2. 編譯與構建主流程

在建立 module 以前,Compiler 會觸發 make,並調用 Compilation.addEntry 方法,經過 options 對象的 entry 字段找到咱們的入口js文件。以後,在 addEntry 中調用私有方法 _addModuleChain ,這個方法主要作了兩件事情。一是根據模塊的類型獲取對應的模塊工廠並建立模塊,二是構建模塊。

而構建模塊做爲最耗時的一步,又可細化爲三步:

  • 調用各 loader 處理模塊之間的依賴

    webpack 提供的一個很大的便利就是能將全部資源都整合成模塊,不只僅是 js 文件。因此須要一些 loader ,好比 url-loader , jsx-loader , css-loader 等等來讓咱們能夠直接在源文件中引用各種資源。webpack 調用 doBuild() ,對每個 require() 用對應的 loader 進行加工,最後生成一個 js module。

    Compilation.prototype._addModuleChain = function process(context, dependency, onModule, callback) {
      var start = this.profile && +new Date();
      ...
      // 根據模塊的類型獲取對應的模塊工廠並建立模塊
      var moduleFactory = this.dependencyFactories.get(dependency.constructor);
      ...
      moduleFactory.create(context, dependency, function(err, module) {
        var result = this.addModule(module);
        ...
        this.buildModule(module, function(err) {
          ...
          // 構建模塊,添加依賴模塊
        }.bind(this));
      }.bind(this));
    };

     

  • 調用 acorn 解析經 loader 處理後的源文件生成抽象語法樹 AST

     Parser.prototype.parse = function parse(source, initialState) {
      var ast;
      if (!ast) {
        // acorn以es6的語法進行解析
        ast = acorn.parse(source, {
          ranges: true,
          locations: true,
          ecmaVersion: 6,
          sourceType: "module"
        });
      }
      ...
    };

     

  • 遍歷 AST,構建該模塊所依賴的模塊

    對於當前模塊,或許存在着多個依賴模塊。當前模塊會開闢一個依賴模塊的數組,在遍歷 AST 時,將 require() 中的模塊經過 addDependency() 添加到數組中。當前模塊構建完成後,webpack 調用 processModuleDependencies 開始遞歸處理依賴的 module,接着就會重複以前的構建步驟。

    Compilation.prototype.addModuleDependencies = function(module, dependencies, bail, cacheGroup, recursive, callback) {
      // 根據依賴數組(dependencies)建立依賴模塊對象
      var factories = [];
      for (var i = 0; i < dependencies.length; i++) {
        var factory = _this.dependencyFactories.get(dependencies[i][0].constructor);
        factories[i] = [factory, dependencies[i]];
      }
      ...
      // 與當前模塊構建步驟相同
    }

     

3. 構建細節

module 是 webpack 構建的核心實體,也是全部 module 的 父類,它有幾種不一樣子類:NormalModule , MultiModule , ContextModule , DelegatedModule 等。但這些核心實體都是在構建中都會去調用對應方法,也就是 build() 。來看看其中具體作了什麼:

// 初始化module信息,如context,id,chunks,dependencies等。
NormalModule.prototype.build = function build(options, compilation, resolver, fs, callback) {
  this.buildTimestamp = new Date().getTime(); // 構建計時
  this.built = true;
  return this.doBuild(options, compilation, resolver, fs, function(err) {
    // 指定模塊引用,不經acorn解析
    if (options.module && options.module.noParse) {
      if (Array.isArray(options.module.noParse)) {
        if (options.module.noParse.some(function(regExp) {
            return typeof regExp === "string" ?
            this.request.indexOf(regExp) === 0 :
              regExp.test(this.request);
          }, this)) {
          return callback();
        }
      } else if (typeof options.module.noParse === "string" ?
        this.request.indexOf(options.module.noParse) === 0 :
          options.module.noParse.test(this.request)) {
        return callback();
      }
    }
    // 由acorn解析生成ast
    try {
      this.parser.parse(this._source.source(), {
        current: this,
        module: this,
        compilation: compilation,
        options: options
      });
    } catch (e) {
      var source = this._source.source();
      this._source = null;
      return callback(new ModuleParseError(this, source, e));
    }
    return callback();
  }.bind(this));
};

 

對於每個 module ,它都會有這樣一個構建方法。固然,它還包括了從構建到輸出的一系列的有關 module 生命週期的函數,咱們經過 module 父類類圖其子類類圖(這裏以 NormalModule 爲例)來觀察其真實形態:

module類圖

能夠看到不管是構建流程,處理依賴流程,包括後面的封裝流程都是與 module 密切相關的。

0.5. 打包輸出

在全部模塊及其依賴模塊 build 完成後,webpack 會監聽 seal 事件調用各插件對構建後的結果進行封裝,要逐次對每一個 module 和 chunk 進行整理,生成編譯後的源碼,合併,拆分,生成 hash 。 同時這是咱們在開發時進行代碼優化和功能添加的關鍵環節。

Compilation.prototype.seal = function seal(callback) {
  this.applyPlugins("seal"); // 觸發插件的seal事件
  this.preparedChunks.sort(function(a, b) {
    if (a.name < b.name) {
      return -1;
    }
    if (a.name > b.name) {
      return 1;
    }
    return 0;
  });
  this.preparedChunks.forEach(function(preparedChunk) {
    var module = preparedChunk.module;
    var chunk = this.addChunk(preparedChunk.name, module);
    chunk.initial = chunk.entry = true;
    // 整理每一個Module和chunk,每一個chunk對應一個輸出文件。
    chunk.addModule(module);
    module.addChunk(chunk);
  }, this);
  this.applyPluginsAsync("optimize-tree", this.chunks, this.modules, function(err) {
    if (err) {
      return callback(err);
    }
    ... // 觸發插件的事件
    this.createChunkAssets(); // 生成最終assets
    ... // 觸發插件的事件
  }.bind(this));
};

 

1. 生成最終 assets

在封裝過程當中,webpack 會調用 Compilation 中的 createChunkAssets 方法進行打包後代碼的生成。 createChunkAssets 流程以下:

createChunkAssets流程

  • 不一樣的 Template

    從上圖能夠看出經過判斷是入口 js 仍是須要異步加載的 js 來選擇不一樣的模板對象進行封裝,入口 js 會採用 webpack 事件流的 render 事件來觸發 Template類 中的 renderChunkModules() (異步加載的 js 會調用 chunkTemplate 中的 render 方法)。

    if(chunk.entry) {
      source = this.mainTemplate.render(this.hash, chunk, this.moduleTemplate, this.dependencyTemplates);
    } else {
      source = this.chunkTemplate.render(chunk, this.moduleTemplate, this.dependencyTemplates);
    }

     

    在 webpack 中有四個 Template 的子類,分別是 MainTemplate.js , ChunkTemplate.jsModuleTemplate.js , HotUpdateChunkTemplate.js ,前二者先前已大體有介紹,而 ModuleTemplate 是對全部模塊進行一個代碼生成,HotUpdateChunkTemplate 是對熱替換模塊的一個處理。

  • 模塊封裝

    模塊在封裝的時候和它在構建時同樣,都是調用各模塊類中的方法。封裝經過調用 module.source()來進行各操做,好比說 require() 的替換。

    MainTemplate.prototype.requireFn = "__webpack_require__";
    MainTemplate.prototype.render = function(hash, chunk, moduleTemplate, dependencyTemplates) {
        var buf = [];
        // 每個module都有一個moduleId,在最後會替換。
        buf.push("function " + this.requireFn + "(moduleId) {");
        buf.push(this.indent(this.applyPluginsWaterfall("require", "", chunk, hash)));
        buf.push("}");
        buf.push("");
        ... // 其他封裝操做
    };

     

  • 生成 assets

    各模塊進行 doBlock 後,把 module 的最終代碼循環添加到 source 中。一個 source 對應着一個 asset 對象,該對象保存了單個文件的文件名( name )和最終代碼( value )。

2. 輸出

最後一步,webpack 調用 Compiler 中的 emitAssets() ,按照 output 中的配置項將文件輸出到了對應的 path 中,從而 webpack 整個打包過程結束。要注意的是,若想對結果進行處理,則須要在 emit 觸發後對自定義插件進行擴展。

0.6. 總結

webpack 的總體流程主要仍是依賴於 compilation 和 module 這兩個對象,但其思想遠不止這麼簡單。最開始也說過,webpack 本質是個插件集合,而且由 tapable 控制各插件在 webpack 事件流上運行,至於具體的思想和細節,將會在後一篇文章中提到。同時,在業務開發中,不管是爲了提高構建效率,或是減少打包文件大小,咱們均可以經過編寫 webpack 插件來進行流程上的控制,這個也會在以後提到。

相關文章
相關標籤/搜索