进阶篇——webpack进阶用法(二)

  上一章节主要介绍了webpack的进阶篇中的自动清理构建、补齐css前缀、静态资源内联。这一节主要介绍多页面应用打包、使用sourcemap、提取页面公共资源。

1.多页面应用打包通用方案

(1)什么是多页面应用

QQ截图20200815160815.png

优势:①每个页面之间都是解耦的;②对于系统的SEO更加友好

(2)打包思路

QQ截图20200815161142.png

上面图是不是很熟悉☺,我们在基础篇-webpack基础用法(三)---html文件压缩里讲过,每个入口文件对应一个html-webpack-plugin。而且entry都是我们配置好的,每增加一个页面我们就要增加一个entry和配置一个html-webpack-plugin,显得过于繁琐,不太友好,有没有更加通用化的方案呢?如下:
QQ截图20200815162029.png

就是动态获取目录文件,使用程序思维自动配置好文件。已上图为例,我们在打包之前必须预定好文件必须放在src目录下面,如:index.js、search.js
npm install glob -D
代码如下

const glob = require('glob');
const path = require('path');

// 设置多页面打包
const setMPA = () => {
    const entry = {};
    const htmlWebpackPlugins = [];

    const entryFiles = glob.sync(path.join(__dirname, './src/*/index.js'));

    Object.keys(entryFiles)
        .map((index) => {
            const entryFile = entryFiles[index];

            const match = entryFile.match(/src\/(.*)\/index\.js/);
            const pageName = match && match[1];

            entry[pageName] = entryFile;
            htmlWebpackPlugins.push(
                new HtmlWebpackPlugin({
                    template: path.join(__dirname, `src/${pageName}/index.html`),
                    filename: `${pageName}.html`,
                    chunks: [pageName],
                    inject: true,
                    minify: {
                        html5: true,
                        collapseWhitespace: true,
                        preserveLineBreaks: false,
                        minifyCSS: true,
                        minifyJS: true,
                        removeComments: false
                    }
                }),
            );
        })

    return {
        entry,
        htmlWebpackPlugins
    }
}

module.exports = {
    entry: entry,
    ...
    
    plugins: [
        new MiniCssExtractPlugin({
            filename: '[name]_[contenthash:8].css'
        }),
        new OptimizeCSSAssetsPlugin({
            assetNameRegExp: /\.css$/g,
            cssProcessor: require('cssnano')
        }),
        
        new CleanWebpackPlugin()
    ].concat(htmlWebpackPlugins)
};

这样我们每次添加页面时,就不用了手动去修改webpack配置了。

2.使用sourceMap

我们在使用webpack进行打包的时候,它会把我们的代码打包成bundle文件,但是在打包的过程中可能会出现错误,我们会发现控制台会把一大串的错误代码打印出来,但是不会定位到具体的错误文件,这时就要轮到我们的sourcemap登场了,具体作用如下

QQ截图20200817202604.png

(1)关键字
QQ截图20200817202757.png
(2)类型
QQ截图20200817203040.png
① eval类型的sourcemap
QQ截图20200817203605.png

npm run build打包之后,我们打开dist文件下的js文件如下


QQ截图20200817203736.png

QQ截图20200817204019.png

我们可以看到,eval把我们的js文件打包之后用eval()包裹了起来,后面用sourceURL制定文件路径。

② source map类型
QQ截图20200817204301.png

导报之后


QQ截图20200817204456.png

可以看到,它把我们的js文件进行了分离,生成了一个js文件一个.map文件。我们打开js文件,拉倒最后一行,会出现如下一句话


QQ截图20200817204718.png

这表示,该js文件要使用的是哪一个map文件。
③ inline-source-map类型

QQ截图20200817204930.png

打包之后
QQ截图20200817205053.png

可以看到,打包之后的dist文件目录下已经没有.map文件了,那么它跑到哪里去了呢?
我们打开js文件,拉倒最后一行
QQ截图20200817205335.png

直接使用sourceMappingURL给引进来了,但是我们也会发现这个js也会变大许多。

④ 开发调试

打开webpack.dev.js文件,如下代码

'use strict';

const glob = require('glob');
const path = require('path');
const webpack = require('webpack');
const HtmlWebpackPlugin = require('html-webpack-plugin');
const { CleanWebpackPlugin } = require('clean-webpack-plugin');

// 设置多页面打包
const setMPA = () => {
    const entry = {};
    const htmlWebpackPlugins = [];

    const entryFiles = glob.sync(path.join(__dirname, './src/*/index.js'));

    Object.keys(entryFiles)
        .map((index) => {
            const entryFile = entryFiles[index];

            const match = entryFile.match(/src\/(.*)\/index\.js/);
            const pageName = match && match[1];

            entry[pageName] = entryFile;
            htmlWebpackPlugins.push(
                new HtmlWebpackPlugin({
                    template: path.join(__dirname, `src/${pageName}/index.html`),
                    filename: `${pageName}.html`,
                    chunks: [pageName],
                    inject: true,
                    minify: {
                        html5: true,
                        collapseWhitespace: true,
                        preserveLineBreaks: false,
                        minifyCSS: true,
                        minifyJS: true,
                        removeComments: false
                    }
                }),
            );
        })

    return {
        entry,
        htmlWebpackPlugins
    }
}

const { entry, htmlWebpackPlugins } = setMPA();

module.exports = {
    entry: entry,
    output: {
        path: path.join(__dirname, 'dist'),
        filename:  '[name].js'
    },
    mode: 'development',
    module: {
        rules: [
            {
                test: /.js$/,
                use: 'babel-loader'
            },
            {
                test: /.css$/,
                use: [
                    'style-loader',
                    'css-loader'
                ]
            },
            {
                test: /.less$/,
                use: [
                    'style-loader',
                    'css-loader',
                    'less-loader'
                ]
            },
            {
                test: /.(png|jpg|gif|jpeg)$/,
                use: [
                    {
                        loader: 'url-loader',
                        options: {
                            limit: 10240
                        }
                    }
                ]
            },
            {
                test: /.(woff|woff2|eot|ttf|otf)$/,
                use: 'file-loader'
            }
        ]
    },
    plugins: [
        new webpack.HotModuleReplacementPlugin(),
        new CleanWebpackPlugin()
    ].concat(htmlWebpackPlugins),
    devServer: {
        contentBase: './dist',   // 打印目录下的信息
        hot: true     // 开启热更新
    },
    devtool: 'source-map'
};

运行
npm run dev

QQ截图20200817210658.png

可以看到这样调试的话,就能看到源代码了,而且非常方便调试。

⑤ cheap-source-map类型
QQ截图20200817211017.png

在js文件下故意写错一行代码,如下

QQ截图20200817211051.png

QQ截图20200817211408.png

可以看到,cheap-source-map帮我们定位了错误在哪一个文件哪一行,就会方便开发调试了。

3.提取页面公共资源

我们在项目开发中,可能各个页面中使用同一套的资源库,或者各个页面也有可能使用相同的css样式,如果我们在打包的时候,就会把这些页面中的资源都会打包一份,这样就会降低打包速度、浪费打包资源、是项目的整体资源过大等等,因此我们需要将这些相同资源提取到一个公共文件中,降低体积。这里我们一react为例。

(1)基础库分离

QQ截图20200818203456.png

entry:可以使第三链接,也可以是本地资源地址

(2)利⽤ SplitChunksPlugin 进⾏公共脚本分离

QQ截图20200818203636.png

minSize(字节):抽离的公共包最小的大小;
maxSize(字节):抽离的公共包最大的大小;
minChunks:表示一个方法在项目中使用的最低次数,如果设置成2,那就表示这个方法在项目最低使用2次,webpack才会将这个方法提取成公共方法;
maxAsyncRequests:浏览器每次请求一步资源的次数。比如浏览器在异步请求一个js时的次数超过指定的值时,就会提取成公共资源;

(3)利⽤ SplitChunksPlugin 分离基础包
QQ截图20200818204824.png
(4)利⽤ SplitChunksPlugin 分离⻚⾯公共⽂件
QQ截图20200818205004.png
(5)实战演练
① html-webpack-externals-plugin

安装 html-webpack-externals-plugin插件
npm install html-webpack-externals-plugin -D

QQ截图20200818210449.png

在html中引入react、react-dom资源
QQ截图20200818211549.png

构建一下


QQ截图20200818210602.png

对比以前的打包方式,发现体积差不多小了100K左右。

② SplitChunksPlugin分离
QQ截图20200818212009.png

构建如下


QQ截图20200818212133.png
③ SplitChunksPlugin分离⻚⾯公共⽂件

我们在项目的根目录下创建一个文件,export一个函数,在不同的页面分别引入一下,配置一下webpack,如下


QQ截图20200818214927.png

构建一下项目


QQ截图20200818215026.png

会把公共方法放在了common.js文件里面

总结

主要学习了多页面应用打包通用方案、使用sourceMap、提取页面公共资源这三个方面的指示点,下一节学习tree shaking、scope hoisting、动态分割和import等方法。
来源极客时间

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342