webpack学习笔记

2021-01-07  本文已影响0人  BigTooth_3611
  1. 命令窗口运行初始化eslint
    切换到./node_modules/.bin/文件夹下运行 eslint --init
    用airbnb

fix: true可以自动修复问题

// eslint-disable-next-line 不转义

"eslintConfig": {
    "extends": "airbnb-base",
    "env": {
        es6: true, // 启用 ES6 语法支持以及新的 ES6 全局变量或类型
        node: true, // Node.js 全局变量和 Node.js 作用域
        browser: true, // 浏览器全局变量
        jquery: true // jQuery 全局变量
    }
}

webpack4.0以上的版本都要安装webpack-cli,否则报错

  1. js兼容

babel @bable/core

  1. HMR
if(module.hot) {
    module.hot.accpet('./print.js', function() {
        console.log('Accepting the updated printMe module!');
        printMe();
    })
}

只有print.js会被重新加载,其他文件不会被重新加载
注意:HMR功能对js的处理,只能处理非入口js文件

4.source-map
提供了源代码到构造后代码映射关系(如果构建后代码出错了,通过映射可以追踪源代码错误)

devtool: 'source-map' 
  1. 运行npm run dev遇到的问题:Error: Cannot find module 'webpack-cli/bin/config-yargs'
  1. oneof 只处理一个loader
  1. 缓存
  1. tree shaking 去除无用代码
  1. 代码分隔
entry: {
    index: './src/js/index.js',
    test: './src/js/test.js'
}
optimization: {
    splitChunks: {
        chunks: 'all'
    }
}
import (/* webpackChunkName: 'test'*/'./test').then(() => {
    console.log('success');
}).catch(() => {
    console.log('error');
})
  1. 懒加载
button.onclick = () => import(/*webpackChunkName: 'print', webpackPrefetch: true*/'./print').then((module) => {
    const print = module.default;
    print();
});
  1. PWA 渐进式网络开发应用程序(离线可访问)

    workbox --> workbox-webpack-plugin
new workboxWebpackPugin.GenerateSw({
    /*
        1.帮助serviceworker快速启动
        2.删除旧的serviceworker
        生成一个serviceworker配置文件~
    */
    clientClaim: true,
    skipWaiting: true
})

入口文件index.js

/*
  1.eslint不认识window、navigator全局变量
  解决:需要修改package.json中eslintConfig配置
  "env": {
      "browser": true // 支持浏览器端全局变量
  }
  2.serviceworker代码必须运行在服务器上
  --> nodejs
  --> npm i serve -g
    serve -s build 启动服务器,将build目录下所有资源作为静态资源暴露出去
*/
// 注册serviceworker
// 处理兼容问题
if('serviceWorker' in navigator) {
    window.addEventListener('load', () => {
        navigator.serviceWorker.register('./service-worker.js')
        .then(() => {
            console.log('serviceworker注册成功');
        })
        .catch(() => {
            console.log('注册失败');
        })
    })
}
  1. 多进程打包
{
    test: /\.js$/,
    exclude: /node_modules/,
    use: [
        /* 
            开启多进程打包
            进程启动大概为600ms,进程通信也有开销
            只有工作消耗时间比较长,才需要多进程打包
        */
        {
            loader: 'thread-loader',
            options: {
                worker: 2 // 进程2个
            }
        },
        {
            loader: 'babel-loader',
            options: [
                presets: [
                    [
                        '@babel/preset-env',
                        {
                            useBuiltIns: 'usage',
                            corejs: {version: 3},
                            targets: {
                                chrome: '60',
                                firefox: '50'
                            }
                        }
                    ]
                ]
            ]
        }
    ]
}
  1. externals
externals: {
    // 拒绝打包jquery
    jquery: 'jQuery'
}
  1. dll 优化重复打包第三方库,加快打包速度
const resolve = require('path');
const webpack = require('webpack');
module.exports = {
    entry: {
        // 最终打包生成的[name] --> jquery
        // ['jquery'] --> 要打包的库是jquery
        jquery: ['jquery']
    },
    output: {
        filename: '[name].js',
        path: resolve(__dirname, 'dll'),
        library: '[name]_[hash]', // 打包的库里面向外暴露出去的内空叫什么名字
    },
    plugins: [
        // 打包生成一个manifest.json --> 提供和jquery映射
        new webpack.DllPlugin({
            name: '[name]_[hash]', // 映射库暴露的内容名称
            path: resolve(__dirname, 'dll/manifest.json') // 输出文件路径
        })
    ]
}

生成dll文件后,就不需要打包了

const resolve = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');
const webpack = require('webpack');
const AddAssetHtmlWebpackPlugin = require('add-asset-html-webpack-plugin');
module.exports = {
    entry: './src/index.js',
    output: {
        filename: 'build.js',
        path: resolve(__dirname, 'build')
    },
    plugins: [
        new HtmlWebpackPlugin({
            template: './src/index.html'
        }),
        // 告诉webpack哪些库不参与打包,同时使用时的名称也得变
        new webpack.DllRefrencePlugin({
            manifest: resolve(_dirname,'dll/manifest.json')
        }),
        // 将某个文件打包输出去,并在html中自动引入该资源
        new AddAssetHtmlWebpackPlugin({
            filepath: resolve(__dirname, 'dll/jquery.js')
        })
    ]
}

  1. 性能优化总结
  1. entry
const {resolve} = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');

/*
    entry: 入口起点
    1. string --> './src/index.js' 
        单入口
        打包形成一个chunk,输出一个bundle文件
        filename: '[name].js' 此时chunk的名称默认是main
    2. array --> ['./src/index.js', './src/add.js']
        多入口
        所有入口文件最终只会形成一个chunk,输出出去只有一个bundle文件
        --> 只有在HMR功能中让html热更新生效~
    3. object --> {index: './src/index.js', add: './src/add.js'}
        多入口
        有几个入口文件就形成几个chunk,输出几个bundle文件
        此时chunk的名称是key
        --> 特殊用法
        {
            // 所有入口文件最终只会形成一个chunk,输出出去只有一个bundle文件
            index: ['./src/index.js', './src/count.js'], 
            // 形成一个chunk,输出一个bundle文件
            add: './src/add.js'
        }
*/
module.exports = {
    entry: './src/index.js',
    output: {
        filename: '[name].js',
        path: resolve(__dirname, 'build')
    }
}
  1. output
const {resolve} = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
    entry: './src/index.js',
    output: {
        // 文件名称(指定名称+目录)
        filename: '[name].js',
        // 输出文件目录(将来所有资源输出的公共目录)
        path: resolve(__dirname, 'build'),
        // 所有资源引入公共路径的前缀 --> 'imgs/a.jpg' --> './imgs/a.jpg'
        publicPath: '/', // 一般用于生产环境
        chunkFilename: '[name]_chunk.js', // 非入口chunk的名称
        // 通常结合dll用
        library: '[name]', // 整个库向外暴露的变量名
        libraryTarget: 'window', // 变量名添加到哪个上 browser
        libraryTarget: 'global', // 变量名添加到哪个上 node
    }
}

index.js

import count from './count';

console.log('index.js文件加载~');

/*
    chunkFilename: '[name]_chunk.js'
    --> 使用:打包成0_chunk.js(0为id)
    --> 不使用:使用filename: '[name].js'规则,打包成0.js
*/
import('./add').then(({default: add}) => {
    console.log(add(1,2);
})

console.log(count(3,2));
  1. loader
const {resolve} = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
    entry: './src/index.js',
    output: {
        filename: '[name].js',
        path: resolve(__dirname, 'build'),
    },
    module: {
        rules: [
            {
                // loader的配置
                test: /\.css$/,
                // 多个loader用use
                use: ['style-loader', 'css-loader'],
            },
            {
                test: /\.js$/,
                // 排除node_modules下的js文件
                exclude: /node_modules/,
                // 只检查src下的js文件
                include: resolve(__dirname, 'src'),
                // 优先执行
                enforce: 'pre',
                // 延后执行
                enforce: 'post',
                // 单个loader用loader
                loader: 'eslint-loader',
                options: {}
            },
            {
                // 以下配置只会生效一个
                oneOf: []
            }
        ],
    }
}
  1. resolve
const {resolve} = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
    entry: './src/index.js',
    output: {
        filename: '[name].js',
        path: resolve(__dirname, 'build'),
    },
    module: {
        rules: [
            {
                // loader的配置
                test: /\.css$/,
                // 多个loader用use
                use: ['style-loader', 'css-loader'],
            },
            {
                test: /\.js$/,
                // 排除node_modules下的js文件
                exclude: /node_modules/
            }
        ]
    },
    mode: 'development',
    // 解析模块的规则
    resolve: {
        // 配置解析模块路径别名:简写路径 缺点路径没有提示
        alias: {
            $css: resolve(__dirname, 'src/css')
        },
        // 配置省略文件路径的后缀名
        extensions: ['.js', '.json', '.jsx'],
        // 告诉webpack解析模块是去找哪个目录
        modules: [resolve(__dirname, '../../node_modules'), 'node_modules']
    }
}

index.js

import '../css/index.css'
--> 可以写成: import '$css/index.css'
  1. devServer
const {resolve} = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
    entry: './src/index.js',
    output: {
        filename: '[name].js',
        path: resolve(__dirname, 'build'),
    },
    module: {
        rules: [
            {
                // loader的配置
                test: /\.css$/,
                // 多个loader用use
                use: ['style-loader', 'css-loader'],
            },
            {
                test: /\.js$/,
                // 排除node_modules下的js文件
                exclude: /node_modules/
            }
        ]
    },
    devServer: {
        // 运行代码的目录
        contentBase: resolve(__dirname, 'build'),
        // 监视contentBase目录下的所有文件,一旦文件变化就会reload
        watchContentBase: true,
        watchOptions: {
            // 忽略文件
            ignored: /node_modules/
        },
        // 启动gzip压缩
        compress: true,
        // 端口号
        port: 5000,
        // 域名
        host: 'localhost',
        // 自动打开浏览器
        open: true,
        // 开启HMR功能
        host: true,
        // 不显示启动服务器日志信息
        clientLogLevel: 'none',
        // 除了一些基本启动信息以外,其他内容都不要显示
        quiet: true,
        // 如果出错了,不要全屏提示~
        overlay: false,
        // 服务器代理 --> 解决开发环境跨域问题
        proxy: {
            // 一旦devServe(5000)服务器受到/api/xx的请求,就会把请求文件转到另外一服务器(3000)
            '/api': {
                target: 'http://localhost:3000',
                // 发送请求时,请求路径重写:将/api/xxx --> /xxx (去掉/api)
                pathRewirte: {
                    '^/api': ''
                }
            }
        }
    }
}

浏览器和服务器是存在跨域问题,服务器和服务器不存在跨域问题,浏览器通过代理服务器向服务器发送请求,代理服务器把请求数据返回到代理服务器上

23.optimization

const {resolve} = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');
const TerserWebpackPlugin = require('terser-webpack-plugin');

module.exports = {
    entry: './src/index.js',
    output: {
        filename: '[name].js',
        path: resolve(__dirname, 'build'),
    },
    module: {
        rules: [
            {
                // loader的配置
                test: /\.css$/,
                // 多个loader用use
                use: ['style-loader', 'css-loader'],
            },
            {
                test: /\.js$/,
                // 排除node_modules下的js文件
                exclude: /node_modules/
            }
        ]
    },
    mode: 'production',
    optimization: {
        splitChunks: {
            chunks: 'all',
            // 以下是默认值,可以不写
            /*
            minSize: 30*1024, // 分割的chunk最小为30kb
            maxSize: 0, // 最大没有限制
            minChunks: 1, // 要提取的chunk最少被引用1次
            maxAsyncRequests: 5, // 按需加载时并行加载的文件的最大数量
            maxInitialRequests: 3, // 入中js文件最大并行请求数量
            automaticNameDelimiter: '~', // 名称连接符 vendors~xxx.js 因为这个里是~
            name: true, //可以使用命名规则
            cacheGroups: { // 分割chunk的组
                // node_modules文件会被打包到vendors --> vendors~xxx.js xxx是模块名称
                // 满足上面的公共规则:如:大小超过30kb,至少被引用一次
                vendors: { // vendors~xxx.js 因为这个里是vendors
                    test: /[\\/]node_modules[\\/]/,
                    // 打包的优先级
                    priority: -10
                },
                default: {
                    // 要提取的chunk最少被引用2次
                    minChunks: 2,
                    // 优先级
                    priority: -20,
                    // 如果当前要打包的模块,和之前已经被提取的模块是同一个,就会复用,而不是重新打包模块
                    reuseExistingChunk: true
                }
            }
            */
        },
        // 将当前模块的记录其他模块的的hash单独打包为一个文件runtime
        // 解决:修改a文件导致b文件contenthash变化
        runtimeChunk: {
            name: entrypoint => `runtime-${entrypoint.name}`
        },
        minimizer: {
            // 配置生产环境的压缩方案:js和css
            new TerserWebpackPlugin: {
                // 开启缓存
                cache: true,
                // 开启多进程打包
                parallel: true,
                // 启动source-map
                sourceMap: true
            }
        }
    }
}
上一篇 下一篇

猜你喜欢

热点阅读