亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

詳解webpack 多頁面/入口支持&公共組件單獨打包

 更新時間:2017年06月29日 15:01:33   作者:sloong  
這篇文章主要介紹了詳解webpack 多頁面/入口支持&公共組件單獨打包,具有一定的參考價值,感興趣的小伙伴們可以參考一下

webpack系列目錄

webpack 系列 二:webpack 介紹&安裝

webpack 系列 三:webpack 如何集成第三方js庫

webpack 系列 四:webpack 多頁面支持 & 公共組件單獨打包

webpack 系列 五:webpack Loaders 模塊加載器

webpack 系列 六:前端項目模板-webpack+gulp實現(xiàn)自動構建部署

基于webpack搭建純靜態(tài)頁面型前端工程解決方案模板, 最終形態(tài)源碼見github: https://github.com/ifengkou/webpack-template

正文

本篇主要介紹:如何自動構建入口文件,并生成對應的output;公共js庫如何單獨打包。

多入口文件,自動掃描入口。同時支持SPA和多頁面型的項目

公共js庫如何單獨打包。

上一篇示例,主要介紹如何集成第三方js庫到項目中使用,如jquery。示例的入口只有一個index,而且是將公共js庫連同page.js一起打包到output.js中。那么在開發(fā)中會出現(xiàn),每新增一個頁面模塊,就需要修改webpack.config.js配置文件(增加一個入口),而且如果用到的第三方庫比較多,這樣也容易導致jquery,React等代碼庫重復被合并到打包后的js,導致js體積過大,頁面加載時間過長

基礎結構和準備工作

以下示例基于上一篇進行改進,上一篇項目源碼

目錄結構說明

.
├── package.json    # 項目配置
├── src      # 源碼目錄
│ ├── pageA.html    # 入口文件a
│ ├── pageB.html    # 入口文件b
│ ├── css/     # css資源
│ ├── img/     # 圖片資源
│ ├── js     # js&jsx資源
│ │ ├── pageA.js    # a頁面入口
│ │ ├── pageB.js    # b頁面入口
│ │ ├── lib/    # 沒有存放在npm的第三方庫或者下載存放到本地的基礎庫,如jQuery、Zepto、avalon
│ ├── pathmap.json   # 手動配置某些模塊的路徑,可以加快webpack的編譯速度
├── webpack.config.js   # webpack配置入口

一:自動構建入口

官方多入口示例

webpack默認支持多入口,官方也有多入口的示例。配件文件webpack.config.js如下

//已簡化
var path = require("path");
module.exports = {
 entry: {
  pageA: "./pageA",
  pageB: "./pageB"
 },
 output: {
  path: path.join(__dirname, "js"),
  filename: "[name].bundle.js",
  chunkFilename: "[id].chunk.js"
 }
}

每新增一個頁面就需要在webpack.config.js的entry 中增加一個 pageC:"./pageC",頁面少還好,頁面一多,就有點麻煩了,而且配置文件,盡可能不改動。那么如何支持不修改配置呢?

自動構建入口函數(shù)

entry實際上是一個map對象,結構如下{filename:filepath},那么我們可以根據(jù)文件名匹配,很容易構造自動掃描器:
npm 中有一個用于文件名匹配的 glob模塊,通過glob很容易遍歷出src/js目錄下的所有js文件:

安裝glob模塊

$ npm install glob --save-dev

修改webpack.config.js 配置,新增entries函數(shù),修改entry:entries(),修改output的filename為"[name].js"

//引入glob
var glob = require('glob')
//entries函數(shù)
var entries= function () {
 var jsDir = path.resolve(srcDir, 'js')
 var entryFiles = glob.sync(jsDir + '/*.{js,jsx}')
 var map = {};

 for (var i = 0; i < entryFiles.length; i++) {
  var filePath = entryFiles[i];
  var filename = filePath.substring(filePath.lastIndexOf('\/') + 1, filePath.lastIndexOf('.'));
  map[filename] = filePath;
 }
 return map;
}
//修改入口,已經(jīng)修改outp的filename
module.exports = {
 //entry: "./src/js/index.js",
 entry: entries(),
 output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
 },
 ......
 //以下省略,可以見下文詳細配置

測試

1.在src/js目錄中新增pageA.js

//js只有兩行代碼,在body中加一句話
var $ = require("jquery")
$("<div>這是jquery生成的多頁面示例</div>").appendTo("body")

2.新增pageA.html,也順便修改原來的index.html 對于js文件名的更改

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
</head>
<body>
<script src="../dist/index.js"></script>
</body>
</html>

3.執(zhí)行webpack,啟動dev-server

$ webpack
$ webpack-dev-server

OK,成功打包生成pageA.js,成功運行

二:公共庫單獨打包

先來分析下,上個步驟打包的日志:

index.js 依賴了avalon 和 jquery,然后打包后的index.js 有480kb

pageA.js 只用了jquery,然后打包后的js 有294kb

那么如果引用的lib庫多一點,又被很多頁面引用,那么lib庫就會被重復打包到page.js中去,模塊越多重復加載的情況越嚴重。

如果把公共代碼提取出來作為單獨的js,那么就到處可以復用,瀏覽器也就可以進行緩存,這時候就需要用到webpack內置插件WebPack.optimize.CommonsChunkPlugin

CommonsChunkPlugin 介紹

使用

new webpack.optimize.CommonsChunkPlugin(options)

Options
翻譯得比較簡單,詳見官方說明:

  1. options.name or options.names(string|string[]): 公共模塊的名稱
  2. options.filename (string): 公開模塊的文件名(生成的文件名)
  3. options.minChunks (number|Infinity|function(module,count) - boolean): 為number表示需要被多少個entries依賴才會被打包到公共代碼庫;為Infinity 僅僅創(chuàng)建公共組件塊,不會把任何modules打包進去。并且提供function,以便于自定義邏輯。
  4. options.chunks(string[]):只對該chunks中的代碼進行提取。
  5. options.children(boolean):如果為true,那么公共組件的所有子依賴都將被選擇進來
  6. options.async(boolean|string):如果為true,將創(chuàng)建一個 option.name的子chunks(options.chunks的同級chunks) 異步common chunk
  7. options.minSize(number):所有公共module的size 要大于number,才會創(chuàng)建common chunk

2個常用的例子,更多例子見官方說明:

1.Commons chunk for entries:針對入口文件提取公共代碼

new CommonsChunkPlugin({
 name: "commons",
 // (the commons chunk name)

 filename: "commons.js",
 // (the filename of the commons chunk)

 // minChunks: 3,
 // (Modules must be shared between 3 entries)

 // chunks: ["pageA", "pageB"],
 // (Only use these entries)
})

2.Explicit vendor chunk:直接指定第三方依賴庫,打包成公共組件

entry: {
 vendor: ["jquery", "other-lib"],
 app: "./entry"
}
new CommonsChunkPlugin({
 name: "vendor",

 // filename: "vendor.js"
 // (Give the chunk a different name)

 minChunks: Infinity,
 // (with more entries, this ensures that no other module
 // goes into the vendor chunk)
})

CommonsChunkPlugin使用

基于上篇的項目,參考上面的第二個例子,我們將jquery 和 avalon 提取出來打包成vendor.js

完整的webpack.config.js 如下:

```js 
var webpack = require("webpack");
 var path = require("path");
 var srcDir = path.resolve(process.cwd(), 'src');
 var nodeModPath = path.resolve(__dirname, './node_modules');
 var pathMap = require('./src/pathmap.json');
 var glob = require('glob')
 var CommonsChunkPlugin = webpack.optimize.CommonsChunkPlugin;
 var entries= function () {
 var jsDir = path.resolve(srcDir, 'js')
 var entryFiles = glob.sync(jsDir + '/*.{js,jsx}')
 var map = {};
 for (var i = 0; i < entryFiles.length; i++) {
  var filePath = entryFiles[i];
  var filename = filePath.substring(filePath.lastIndexOf('\/') + 1, filePath.lastIndexOf('.'));
  map[filename] = filePath;
 }
 return map;
}

module.exports = {
 //entry: "./src/js/index.js",
 //entry: entries(),
 entry: Object.assign(entries(), {
  // 用到什么公共lib(例如jquery.js),就把它加進vendor去,目的是將公用庫單獨提取打包
  'vendor': ['jquery', 'avalon']
 }),
 output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
 },
 module: {
  loaders: [
   {test: /\.css$/, loader: 'style-loader!css-loader'}
  ]
 },
 resolve: {
  extensions: ['.js', "", ".css"],
  root: [srcDir,nodeModPath],
  alias: pathMap,
  publicPath: '/'
 },
 plugins: [
  new CommonsChunkPlugin({
   name: 'vendor',
   minChunks: Infinity
  })
 ]
}
```

測試、驗證

1.修改入口(Object.assign 是html5.js里面的....)

//entry: entries(),
entry: Object.assign(entries(), {
 // 用到什么公共lib(例如jquery.js),就把它加進vendor去,目的是將公用庫單獨提取打包
 'vendor': ['jquery', 'avalon']
}),

2.加入插件CommonsChunkPlugin

var CommonsChunkPlugin = webpack.optimize.CommonsChunkPlugin;
config 中增加 plugins
 plugins: [
  new CommonsChunkPlugin({
   name: 'vendor',
   minChunks: Infinity
  })
 ]

3.修改index.html 和 pageA.html,增加對verdor.js的引用

<script src="../dist/vendor.js"></script>
<script src="../dist/index.js"></script>
//<script src="../dist/pageA.js"></script>

4.執(zhí)行webpack

$ webpack

結果分析

可以看到index.js 就只有457 bytes了,pageA.js 227bytes。vendor.js 是集成了jquery+avalon,所以有488kb。

這樣vendor.js 就可以重復利用了,也方便瀏覽器進行緩存。

調試過程中發(fā)現(xiàn)

Uncaught ReferenceError: webpackJsonp is not defined

這個是因為當時把vendor.js引入 放到了page.js 后面,導致page.js執(zhí)行異常,所以,請一定把vendor.js 放在前面。

生成后的index.js就很輕便了,第三方庫都被打包到vendor中了,代碼如下:

webpackJsonp([0],[
/* 0 */
/***/ function(module, exports, __webpack_require__) {

 /**
  * Created by sloong on 2016/6/1.
  */
 //avalon 測試
 var avalon = __webpack_require__(1);
 avalon.define({
  $id: "avalonCtrl",
  name: "Hello Avalon!"
 });

 /*
 //zepto 測試
 require("zepto")

 $("<div>這是zepto生成的</div>").appendTo("body")*/

 //jquery 測試
 var $ = __webpack_require__(2)
 $("<div>這是jquery生成的</div>").appendTo("body")

/***/ }
]);

頁面測試均正常

OK,本篇結束了。如何讓webpack 自動在html文件中引入所需js的script標簽,如何給js和css文件加了hash值,這樣瀏覽器每次都能檢測到文件變更,而且也不需要手動修改引入的js文件鏈接,這些操作webpack都能輕松搞定

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • js的壓縮及jquery壓縮探討(提高頁面加載性能/保護勞動成果)

    js的壓縮及jquery壓縮探討(提高頁面加載性能/保護勞動成果)

    搞定js的加密和壓縮,一方面可以提高頁面加載性能,另外一方面也希望辛苦研發(fā)出來的成果得到一定的保護,感興趣的朋友可以了解下,或許對你有所幫助
    2013-01-01
  • JavaScript練習小項目之修改div塊的顏色

    JavaScript練習小項目之修改div塊的顏色

    這篇文章主要給大家介紹了關于JavaScript練習小項目之修改div塊的顏色的相關資料,文中通過舉例介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2023-01-01
  • 原生JS實現(xiàn)的放大鏡特效示例【測試可用】

    原生JS實現(xiàn)的放大鏡特效示例【測試可用】

    這篇文章主要介紹了原生JS實現(xiàn)的放大鏡特效,涉及javascript事件響應及頁面元素動態(tài)操作相關實現(xiàn)技巧,需要的朋友可以參考下
    2018-12-12
  • 基于javascript原生判斷DOM是否加載完畢

    基于javascript原生判斷DOM是否加載完畢

    這篇文章主要介紹了基于javascript原生判斷DOM是否加載完畢,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-10-10
  • 用file標簽實現(xiàn)多圖文件上傳預覽

    用file標簽實現(xiàn)多圖文件上傳預覽

    本文介紹了用file標簽實現(xiàn)多圖文件上傳預覽的方法。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-02-02
  • js 實現(xiàn) input type="file" 文件上傳示例代碼

    js 實現(xiàn) input type="file" 文件上傳示例代碼

    在開發(fā)中,文件上傳必不可少但是它長得又丑、瀏覽的字樣不能換,一般會讓其隱藏點其他的標簽(圖片等)來時實現(xiàn)選擇文件上傳功能
    2013-08-08
  • 最全的package.json解析

    最全的package.json解析

    從我們接觸前端開始,每個項目的根目錄下一般都會有一個package.json文件,這個文件定義了當前項目所需要的各種模塊,以及項目的配置信息,本文就詳細的來介紹一下
    2021-07-07
  • JavaScript中引用vs復制示例詳析

    JavaScript中引用vs復制示例詳析

    這篇文章主要給大家介紹了關于JavaScript中引用vs復制的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-12-12
  • JS實現(xiàn)頁面跳轉與刷新的方法匯總

    JS實現(xiàn)頁面跳轉與刷新的方法匯總

    這篇文章主要給大家介紹了關于JS實現(xiàn)頁面跳轉與刷新的方法,文中通過示例代碼介紹的非常詳細,對大家學習或者使用JS具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧
    2019-08-08
  • JS實現(xiàn)動態(tài)給標簽控件添加事件的方法示例

    JS實現(xiàn)動態(tài)給標簽控件添加事件的方法示例

    這篇文章主要介紹了JS實現(xiàn)動態(tài)給標簽控件添加事件的方法,結合實例形式分析了javascript簡單實現(xiàn)動態(tài)添加事件的相關操作技巧,需要的朋友可以參考下
    2017-05-05

最新評論