webpack v4 從dev到prd

目錄

  • 概述
  • Big changes
  • 加載loader方法總結(jié)
  • 開發(fā)必備的loader&plugins
  • 優(yōu)化向prd進(jìn)發(fā)
  • 未完待續(xù)

概述

本月迎來了 v4 正式版的發(fā)布,本文用于學(xué)習(xí)新特性和總結(jié)開發(fā)必用plugin & loader,從dev到prd,走你~

Big changes

  • Environment
    • Node.js 4 is no longer supported. Source Code was upgraded to a higher ecmascript version.
  • Usage
    • You have to choose (mode or --mode) between two modes now: production or development

本次新版本中引入了 mode 配置項(xiàng),開發(fā)者可在 none,development(開發(fā) ) 以及 production(產(chǎn)品)三種模式間選擇。該配置項(xiàng)缺省情況下默認(rèn)使用 production 模式。

  • development 模式給你極致的開發(fā)體驗(yàn),包含瀏覽器調(diào)試相關(guān)工具,極快的增量編譯,豐富全面的報(bào)錯(cuò)信息...
  • production 模式則包含大量發(fā)版優(yōu)化,代碼壓縮,絲般潤滑的運(yùn)行時(shí)優(yōu)化,開發(fā)相關(guān)代碼的排除,易用,etc.
  • none 不使用預(yù)設(shè),等于老版本中全部自己配置的原始狀態(tài)。

eg:

webpack --mode development
  • Usage
    • Some Plugin options are now validated
    • CLI has been move to webpack-cli, you need to install webpack-cli to use the CLI
    • The ProgressPlugin (--progress) now displays plugin names
      At least for plugins migrated to the new plugin system

新版中將 webpack 命令行工具拆分到單獨(dú)的倉庫中,所以需要額外安裝 webpack-cli。

npm init -y //初始化項(xiàng)目
npm install webpack webpack-cli -D //安裝webpack webpack-cli 依賴
npx webpack --mode development // npx可以直接運(yùn)行node_modules/.bin目錄下面的命令

或者通過配置package.json的script build
"scripts": {
    "build": "webpack --mode development",
},

加載loader方法總結(jié)

  • use
module: {
    rules:[
        { 
            test: /\.css$/,
            use: ['style-loader','css-loader']
        }
    ]
}

css-loader用來解析處理CSS文件中的url路徑,

要把CSS文件變成一個(gè)模塊
多個(gè)loader是有順序要求的,從右往左寫,因?yàn)檗D(zhuǎn)換的時(shí)候是從右往左轉(zhuǎn)換

此插件先用css-loader處理一下css文件,再用style-loader把CSS文件變成style標(biāo)簽插入head中

  • loader
module: {
    rules:[
        {
            test: /\.css$/,
            loader: ["style-loader", "css-loader"]
        },
    ]
}
  • use+loader
module: {
    rules:[
        {
            test: /\.css$/,
            use:[
                { loader:"style-loader"},
                { 
                    loader: 'css-loader',
                    options: {sourceMap: true}
                }
            ]
        }
    ]
}

這三種loader的寫法,最后打包的結(jié)果相同

loader中的options配置項(xiàng)可以用"?"跟在加載器后面

eg:

{  
    test: /\.jpeg$/,  
    use: 'url-loader?limit=1024&name=[path][name].[ext]&outputPath=img/&publicPath=output/',  
}

為以下配置的簡寫

{  
    test: /\.jpeg$/,  
    use: {
        loader:'url-loader',
        options:{
            limit:1024,
            name:[path][name].[ext],
            outputPath:img/
            publicPath:output/'
        }
    }
}

開發(fā)必備的loader&plugins

  • css-loader
  • babel-loader

講ES6代碼轉(zhuǎn)換為ES5

{
    test: /\.js/,
    use: {
        loader: 'babel-loader',
        query: {
            presets: ["env", "stage-0", "react"]
        }
    }
},

babel-loader的預(yù)設(shè)可以添加在query中,也可以在項(xiàng)目根目錄添加 .babelrc 文件

.babelrc

{
    "presets": [
        "env",
        "stage-0",
        "react"
    ]
}
  • html-webpack-plugin

插件的基本作用就是生成html文件。原理很簡單:

將 webpack中entry配置的相關(guān)入口thunk 和 extract-text-webpack-plugin抽取的css樣式 插入到該插件提供的template或者templateContent配置項(xiàng)指定的內(nèi)容基礎(chǔ)上生成一個(gè)html文件,具體插入方式是將樣式link插入到head元素中,script插入到head或者body中。

const HtmlWebpackPlugin = require('html-webpack-plugin');

new HtmlWebpackPlugin({
    template: './src/index.html',//指定產(chǎn)的HTML模板
    filename: `index.html`,//產(chǎn)出的HTML文件名
    title: 'index',
    hash: true,// 會在引入的js里加入查詢字符串避免緩存,
    minify: {
        removeAttributeQuotes: true
    }
}),

可以用 cnpm search html-webpack-plugin 查找想用loader的用法

  • less-loader sass-loader

優(yōu)化向prd進(jìn)發(fā)

  • 提取公共的css代碼

它會將所有的入口 chunk(entry chunks)中引用的 *.css,移動(dòng)到獨(dú)立分離的 CSS 文件。因此,你的樣式將不再內(nèi)嵌到 JS bundle 中,而是會放到一個(gè)單獨(dú)的 CSS 文件(即 styles.css)當(dāng)中。 如果你的樣式文件大小較大,這會做更快提前加載,因?yàn)?CSS bundle 會跟 JS bundle 并行加載。

npm i extract-text-webpack-plugin@next -D
const ExtractTextWebpackPlugin = require('extract-text-webpack-plugin');
let cssExtract = new ExtractTextWebpackPlugin({
    filename: 'css/css.css',
    allChunks: true
});
module:{
    rules:[
        {
            test: /\.css$/,//轉(zhuǎn)換文件的匹配正則
            loader: cssExtract.extract({
                use: ["css-loader?minimize"]
            })
        },
    ]
}
plugins:[
    ...... ,
    + cssExtract
]
  • 盡量減少文件解析,用resolve配置文件解析路徑,include
rules: {
    test: /\.js$/,
    loader:'babel-loader',
    include: path.resolve(__dirname, 'src'),//只轉(zhuǎn)換或者編譯src 目錄 下的文件
    exclude: /node_modules/ //不要解析node_modules
}
  • resolve.mainFields
WebpackTest
|
|
| - src
|   | - index.js
|
| - lib
|   | - fetch
|       |
|       browser.js
|       node.js
|       package.json
|
| - webpack.config.js

當(dāng)從 npm 包中導(dǎo)入模塊時(shí)(例如,引入lib下的庫),此選項(xiàng)將決定在 package.json 中使用哪個(gè)字段導(dǎo)入模塊。根據(jù) webpack 配置中指定的 target 不同,默認(rèn)值也會有所不同。

package.json

lib文件夾下的package.json中配置相對應(yīng)模塊的key

{
  "name": "fetch",
  "version": "1.0.0",
  "description": "",
  "node": "./node.js",
  "browser": "./browser.js",
  "scripts": {
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "keywords": [],
  "author": "",
  "license": "ISC"
}

webpack.config.js

在resolve解析對象中,加入lib的路徑

resolve: {
    extensions: ['.js', '.json'],
    mainFields: ['main', 'browser', 'node'],
    modules: [path.resolve('node_modules'), path.resolve('lib')]
}

index.js

這樣在index.js中引用第三方庫時(shí),會去查找modules下的路徑中是否配置了所需的文件,知道在package.json中找到mainFields中的key對應(yīng)文件,停止。

let fetch = require('fetch');
console.log(fetch);

打包后 console.log出的對象


image

如果交換mainFields中的key順序

mainFields: ['main', 'node','browser']

打包后 console.log出的對象,因?yàn)檎业搅薻ey=node對應(yīng)的文件就停止了查找

image
  • DllReferencePlugin

這個(gè)插件是在 webpack 主配置文件中設(shè)置的, 這個(gè)插件把只有 dll 的 bundle(們)(dll-only-bundle(s)) 引用到需要的預(yù)編譯的依賴。

新建webpack.react.config.js

const path = require('path');
const webpack = require('webpack')
module.exports = {
    entry: {
        react: ['react', 'react-dom']
    },
    output: {
        path: path.join(__dirname, 'dist'),// 輸出動(dòng)態(tài)連接庫的文件名稱
        filename: '[name]_dll.js',
        library: '_dll_[name]'//全局變量的名字,其它會從此變量上獲取到里面的模塊
    },
    // manifest 表示一個(gè)描述文件
    plugins: [
        new webpack.DllPlugin({
            name: '_dll_[name]',
            path: path.join(__dirname, 'dist', 'manifest.json')//最后打包出來的文件目錄和名字
        })
    ]
}
在entry入口寫入要打包成dll的文件,這里把體積較大的react和react-dom打包

output中的關(guān)鍵是library的全局變量名,下文詳細(xì)說明dll&manifest工作原理

打包dll文件

webpack --config webpack.react.config.js --mode development

打包出來的manifest.json節(jié)選

image

打包出來的react_dll.js節(jié)選

image

可見manifest.json中的 name值就是

output:{
    library:_dll_react
}

manifest.json就是借書證,_dll_react就像圖書館書籍的條形碼,為我們最終找到filename為react_dll.js的參考書

使用“參考書”

在webpack.config.js中加入“借書證”

new webpack.DllReferencePlugin({
    manifest: path.join(__dirname, 'dist', 'manifest.json')
})

再運(yùn)行

webpack --mode development

打包速度顯著變快


image

打包后的main.js中,react,react-dom.js也打包進(jìn)來了,成功~

import React from 'react';\n//import ReactDOM from 'react-dom';
 (function(module, exports, __webpack_require__) {

"use strict";
eval("\n\n//import name from './base';\n//import React from 'react';\n//import ReactDOM from 'react-dom';\n//import ajax from 'ajax';\n//let result = ajax('/ajax');\n\n//ReactDOM.render(<h1>{result}</h1>, document.getElementById('root'));\n// fetch fetch.js fetch.json fetch文件夾\n//let fetch = require('fetch');\n//console.log(fetch);\n//let get = require('../dist/bundle.js');\n//get.getName();\nconsole.log('hello');\n\nvar name = 'zfpx';\nconsole.log(name);\nif (true) {\n    var s = 'ssssssssssssssssssssssss';\n    console.log(s);\n    console.log(s);\n    console.log(s);\n    console.log(s);\n}\n\n//# sourceURL=webpack:///./src/index.js?");

/***/ })

/******/ });

未完待續(xù)

  • webpack.ProvidePlugin
  • 拷貝靜態(tài)資源
  • 壓縮css(npm i -D purifycss-webpack purify-css)

覺得好玩就關(guān)注一下歡迎大家收藏寫評論~~

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • GitChat技術(shù)雜談 前言 本文較長,為了節(jié)省你的閱讀時(shí)間,在文前列寫作思路如下: 什么是 webpack,它要...
    蕭玄辭閱讀 12,892評論 7 110
  • 版權(quán)聲明:本文為博主原創(chuàng)文章,未經(jīng)博主允許不得轉(zhuǎn)載。 webpack介紹和使用 一、webpack介紹 1、由來 ...
    it筱竹閱讀 11,460評論 0 21
  • 原文首發(fā)于:Webpack 3,從入門到放棄 Update (2017.8.27) : 關(guān)于 output.pub...
    昵稱都被用完了衰閱讀 1,994評論 4 19
  • 1206 今天陳信宏42歲啦 一米八的陳信宏真是可愛到爆炸溫柔到爆炸 陳信宏三個(gè)字想想就甜蜜嘻嘻嘻嘻 永遠(yuǎn)愛你永遠(yuǎn)在一起
    木子后花園閱讀 822評論 0 0
  • 因前天在和小伙伴討論怎么用效率手冊,怎么做計(jì)劃,怎么做總結(jié),為什么做了好些天的總結(jié)卻仍然覺得自己的時(shí)間管理...
    lovent閱讀 95評論 0 0

友情鏈接更多精彩內(nèi)容