webpack5新特性

webpack5新特性

  • 持久化缓存
  • 资源模块
  • moduleIds & chunkIds的优化
  • 更智能的tree shaking
  • nodeJs的polyfill脚本被移除
  • 支持生成e6/es2015的代码
  • SplitChunk和模块大小
  • Module Federation

持久化缓存

  • 缓存生成的webpack模块和chunk,来改善构建速度
  • cache 会在开发模式被设置成 type: 'memory' 而且在 生产 模式 中被禁用
  • 在webpack5中默认开启,缓存默认是在内存里,但可以对cache进行设置
  • 当设置cache.type: "filesystem"的时候,webpack会在内部启用文件缓存和内存缓存,写入的时候会同时写入内存和文件,读取缓存的时候会先读内存,如果内存里没有才会读取文件
  • 每个缓存最大资源占用不超过500MB,当逼近或超过500MB时,会优先删除最老的缓存,并且缓存的有效期最长为2周
  • FileMiddleware.js
  • PackFileCacheStrategy.js:1036
  • FileSystemInfo.js:1691
  • 默认情况下,webpack 假定 webpack 所在的 node_modules 目录只被包管理器修改。对 node_modules 来说,哈希值和时间戳会被跳过

安装

js 复制代码
cnpm i webpack webpack-cli webpack-dev-server babel-loader @babel/core  @babel/preset-env -D

webpack.config.js

js 复制代码
const path = require('path');
module.exports = {
    mode: 'development',
    cache: {
        type: 'filesystem',  //  'memory' | 'filesystem'
        cacheDirectory: path.resolve(__dirname, 'node_modules/.cache/webpack'), // 默认将缓存存储在 node_modules/.cache/webpack
    },
    watch: true,
    module: {
        rules: [
            {
                test: /\.js$/,
                use: [
                    {
                        loader: 'babel-loader',
                        options: {
                            presets: [
                                "@babel/preset-env"
                            ]
                        }
                    }
                ]
            }
        ]
    }
}

package.json

json 复制代码
  "scripts": {
    "build": "webpack",
    "debug": "webpack"
  },

资源模块

  • 资源模块(asset module)是一种模块类型,它允许使用资源文件(字体,图标等)而无需配置额外 loader
  • 在 webpack 5 之前,通常使用:
    • raw-loader 将文件导入为字符串
    • url-loader 将文件作为 data URI 内联到 bundle 中
    • file-loader 将文件发送到输出目录
  • 资源模块类型(asset module type),通过添加 4 种新的模块类型,来替换所有这些 loader
    • asset/resource 发送一个单独的文件并导出 URL。之前通过使用 file-loader 实现。
    • asset/inline 导出一个资源的 data URI。之前通过使用 url-loader 实现。
    • asset/source 导出资源的源代码。之前通过使用 raw-loader 实现。
    • asset 在导出一个 data URI 和发送一个单独的文件之间自动选择。之前通过使用 url-loader,并且配置资源体积限制实现
js 复制代码
module.exports = {
  module: {
    rules: [ 
      {
        test: /\.png$/,
        type: 'asset/resource'
      },
      {
        test: /\.ico$/,
        type: 'asset/inline'
      },
      {
        test: /\.txt$/,
        type: 'asset/source'
      }
    ]
  },
  experiments: {
    asset: true
  },
};

老方式

src\index.js

src\index.js

js 复制代码
import url from './images/kf.jpg';
let img = new Image();
img.src = url;
document.body.appendChild(img);

webpack.config.js

webpack.config.js

js 复制代码
module: {
        rules: [
            {
                test: /\.(jpg|png|gif)$/,
                type:'asset'
            }
        ]
}

新方式

  • 新的方式语法是为了允许在没有打包工具的情况下运行代码。这种语法也可以在浏览器中的原生 ECMAScript 模块中使用

src\index.js

src\index.js

js 复制代码
let url = new URL('./images/kf.jpg', import.meta.url);
let img = new Image();
img.src = url;
document.body.appendChild(img);

URIs

  • Webpack 5 支持在请求中处理协议
  • 支持data 支持 Base64 或原始编码,MimeType可以在module.rule中被映射到加载器和模块类型
  • 支持http(s)

src\index.js

js 复制代码
import data from "data:text/javascript,export default 'title'";
import url from 'https://img.zhufengpeixun.com/zfjg.png';
console.log(data,url);

webpack.config.js

js 复制代码
const webpack = require('webpack');
const HtmlWebpackPlugin = require('html-webpack-plugin');
module.exports = {
    target: ['es6'],
    plugins:[
        new webpack.experiments.schemes.HttpsUriPlugin()
    ]
}

moduleIds & chunkIds的优化

概念

  • module: 每一个文件其实都可以看成一个 module
  • chunk: webpack打包最终生成的代码块,代码块会生成文件,一个文件对应一个chunk

优化

  • 在webpack5之前,没有从entry打包的chunk文件,都会以1、2、3...的文件命名方式输出,删除某些些文件可能会导致缓存失效
  • 在生产模式下,默认启用这些功能chunkIds: "deterministic", moduleIds: "deterministic",此算法采用确定性的方式将短数字 ID(3 或 4 个字符)短hash值分配给 modules 和 chunks
可选值 含义 示例
false 不应使用任何内置算法,插件提供自定义算法 Path variable [name] not implemented in this context: [name].js
natural 按使用顺序的数字ID 1
named 方便调试的高可读性id src_two_js.js
deterministic 根据模块名称生成简短的hash值 915
size 根据模块大小生成的数字id 0
diff 复制代码
const path = require('path');
module.exports = {
    mode: 'development',
    devtool:false,
+    optimization:{
+        moduleIds:'deterministic',
+        chunkIds:'size'
+    },
    module: {
        rules: [
            {
                test: /\.js$/,
                use: [
                    {
                        loader: 'babel-loader',
                        options: {
                            presets: [
                                "@babel/preset-env"
                            ]
                        }
                    }
                ]
            }
        ]
    }
}

移除Node.js的polyfill

  • webpack4带了许多Node.js核心模块的polyfill,一旦模块中使用了任何核心模块(如crypto),这些模块就会被自动启用
  • webpack5不再自动引入这些polyfill

安装

js 复制代码
cnpm i crypto-js crypto-browserify stream-browserify buffer -D

src\index.js

js 复制代码
import CryptoJS from 'crypto-js';
console.log(CryptoJS.MD5('zhufeng').toString());

webpack.config.js

js 复制代码
    resolve:{
        /* fallback:{ 
            "crypto": require.resolve("crypto-browserify"),
            "buffer": require.resolve("buffer"),
            "stream":require.resolve("stream-browserify")
        }, */
        fallback:{ 
            "crypto":false,
            "buffer": false,
            "stream":false
        }
    },

更强大的tree-shaking

  • tree-shaking
  • webpack4 本身的 tree shaking 比较简单,主要是找一个 import 进来的变量是否在这个模块内出现过,非常简单粗暴

原理

作用域

  • 而对于一个模块来说,只有 classfunction 的作用域是可以导出到其他模块的
js 复制代码
// module scope start

    // Block

    { // <- scope start
    } // <- scope end

    // Class

    class Foo { // <- scope start
        //   |       
    }       // <- scope end

  // If else

    if (true) { // <- scope start

    } /* <- scope end */ else { // <- scope start

    } // <- scope end

    // For

    for (;;) { // <- scope start
    } // <- scope end

    // Catch

    try {

    } catch (e) { // <- scope start

    } // <- scope end

    // Function

    function() { // <- scope start
    } // <- scope end

    // Scope

    switch() { // <- scope start
    } // <- scope end

// module scope end

开启

开发环境

webpack.config.js

js 复制代码
const path = require('path');

module.exports = {
  entry: './src/index.js',
  output: {
    filename: 'bundle.js',
    path: path.resolve(__dirname, 'dist'),
  },
  mode: 'development',
   optimization: {
   usedExports: true,
  },
};

生产环境

  • 生产环境默认开启

sideEffects

  • "sideEffects": false,意思就是对所有的模块都进行Tree Shaking
  • 也就是将没有引入的方法等不进行打包到打包输出文件中

package.json

json 复制代码
{"sideEffects": ["@babel/polyfill"]}
{"sideEffects": ["*.css"]}

嵌套的 tree-shaking

  • webpack 现在能够跟踪对导出的嵌套属性的访问
  • 这可以改善重新导出命名空间对象时的Tree Shaking(清除未使用的导出和混淆导出)

src\index.js

src\index.js

js 复制代码
import * as calculator from "./calculator";
console.log(calculator.operators.add);

src\calculator.js

src\calculator.js

js 复制代码
import * as operators from "./operators";
export { operators };

src\operators.js

src\operators.js

js 复制代码
export const add = 'add';
export const minus = 'minus';

webpack.config.js

webpack.config.js

js 复制代码
module.exports = {
    mode: 'production'
}

内部模块 tree-shaking

  • webpack 4 没有分析模块的导出和引用之间的依赖关系
  • webpack 5 可以对模块中的标志进行分析,找出导出和引用之间的依赖关系

src\index.js

src\index.js

js 复制代码
import { getPostUrl } from './api';
console.log('getPostUrl',getPostUrl);

src\api.js

src\api.js

js 复制代码
import { host } from './constants';

function useHost() {
  return host;
}

export function getUserUrl() {
  return useHost()+'/user';
}
export function getPostUrl() {
    return '/post';
}

src\api.js

src\api.js

js 复制代码
export const host = 'http://localhost';

CommonJs Tree Shaking

  • webpack 曾经不进行对 CommonJS 导出和 require()调用时的导出使用分析
  • webpack 5 增加了对一些 CommonJS 构造的支持,允许消除未使用的 CommonJs 导出,并从 require() 调用中跟踪引用的导出名称 支持以下构造:
  • exports|this|module.exports.xxx = ...
  • exports|this|module.exports = require("...") (reexport)
  • exports|this|module.exports.xxx = require("...").xxx (reexport)
  • Object.defineProperty(exports|this|module.exports, "xxx", ...)
  • require("abc").xxx
  • require("abc").xxx()

src\index.js

src\index.js

js 复制代码
let api = require('./api');
console.log(api.getPostUrl);

src\api.js

src\api.js

js 复制代码
function getUserUrl() {
  return '/user';
}
function getPostUrl() {
    return '/post';
}

exports.getPostUrl=getPostUrl;

splitChunks

相关推荐
也无晴也无风雨1 小时前
深入剖析输入URL按下回车,浏览器做了什么
前端·后端·计算机网络
Martin -Tang2 小时前
Vue 3 中,ref 和 reactive的区别
前端·javascript·vue.js
FakeOccupational3 小时前
nodejs 020: React语法规则 props和state
前端·javascript·react.js
放逐者-保持本心,方可放逐3 小时前
react 组件应用
开发语言·前端·javascript·react.js·前端框架
曹天骄4 小时前
next中服务端组件共享接口数据
前端·javascript·react.js
阮少年、5 小时前
java后台生成模拟聊天截图并返回给前端
java·开发语言·前端
郝晨妤6 小时前
鸿蒙ArkTS和TS有什么区别?
前端·javascript·typescript·鸿蒙
AvatarGiser6 小时前
《ElementPlus 与 ElementUI 差异集合》Icon 图标 More 差异说明
前端·vue.js·elementui
喝旺仔la6 小时前
vue的样式知识点
前端·javascript·vue.js
别忘了微笑_cuicui6 小时前
elementUI中2个日期组件实现开始时间、结束时间(禁用日期面板、控制开始时间不能超过结束时间的时分秒)实现方案
前端·javascript·elementui