webpack打包优化

项目开始时webpack配置

vue-cli3以后,我们修改webpack配置,需要自己在项目根路径下创建vue.config.js文件。

配置proxy跨域

使用vue-cli开发项目,在本地开发环境中,如果遇到跨域问题。可以通过配置proxy的方式,解决跨域问题

module.exports = {
     
  devServer: {
     
   open: false, // 自动启动浏览器
   host: '0.0.0.0', // localhost
   port: 6060, // 端口号
   hotOnly: false, // 热更新

   overlay: {
     
      //  当出现编译器错误或警告时,在浏览器中显示全屏覆盖层
      warnings: false,
      errors: true
    },
    proxy: {
     
      //配置跨域
      '/api': {
     
        target: 'https://www.test.com', // 接口的域名
        // ws: true, // 是否启用websockets
        changOrigin: true, // 开启代理,在本地创建一个虚拟服务端
        pathRewrite: {
     
          '^/api': '/'
        }
      }
    }
  }
}

配置完成后,当我们在去请求接口时,就可以像下面这样写

this.axios({
     
url:'/api/v1/api/userinfo',
method:'get'
}).then(res=>{
     
//......
})

配置alias别名

使用vue-cli开发项目,最大的特色就是组件化,组件中频繁引用其他组件或插件,我们可以把一些常用的路径定义成简短的名字。方便我们在开发中使用。

//加载path模块
const path = require('path')
//定义resolve方法,把相对路径转换成绝对路径
const resolve = dir => path.join(__dirname, dir)

module.exports = {
     
  chainWebpack: config => {
     
    // 添加别名
    config.resolve.alias
      .set('@', resolve('src'))
      .set('assets', resolve('src/assets'))
      .set('api', resolve('src/api'))
      .set('views', resolve('src/views'))
      .set('components', resolve('src/components'))
  }
}

配置完成之后,我们在项目中可以像下面这样写路径

//之前这么写
import Home from '../views/Home.vue'
//配置alias别名后
import Home from 'views/Home.vue'
//也可以这么写
import Home from '@/views/Home.vue'

项目结束后打包webpack配置

目的:

  • 提高打包速度
  • 减小项目体积,提高首屏加载速度
  • 提高用户体验

项目开发完成后,运行npm run build进行打包操作,打包前对webpack配置

module.exports = {
     
  publicPath: './', // 静态资源路径(默认/,打包后会白屏)
  outputDir: 'dist', // 打包后文件的目录 (默认为dist)
  assetsDir: 'static', //  outputDir的静态资源(js、css、img、fonts)目录  默认为‘’没有单独目录js/css/img在根目录中。
  }

去除生产环境下的sourceMap

vue项目打包之js文件夹中华会自动生成一些map文件,占用相当一部分空间。
sourceMap是资源映射文件,存的是打包前后的代码位置,方便开发使用,
map文件的作用:在项目打包后,代码都是经过压缩的,如果运行时报错,输出的错误信息无法准确得知是哪里的代码错了,有了map就可以像未压缩的代码一样,准确的输出是哪一行的代码有错

生产环境是不需要sourceMap的,下面这步操作可以去除

module.exports = {
     
  //去除生产环境的productionSourceMap
  productionSourceMap: false,
}

使用CDN加速优化

cdn优化是指把第三方库,比如(vue、vue-router、vuex、axios)通过cdn的方式引入项目中,这样vendor.js会显著减少,并且大大的提升项目的首页加载速度,下面是具体操作

const isProduction = process.env.NODE_ENV === 'production';

// externals
const externals = {
     
  vue: 'Vue',
  'vue-router': 'VueRouter',
  vuex: 'Vuex',
  vant: 'vant',
  axios: 'axios'
}
// CDN外链,会插入到index.html中
const cdn = {
     
  // 开发环境
  dev: {
     
    css: [],
    js: []
  },
 // 生产环境
  build: {
     
    css: ['https://cdn.jsdelivr.net/npm/[email protected]/lib/index.css'],
    js: [
      'https://cdn.jsdelivr.net/npm/[email protected]/dist/vue.min.js',
      'https://cdn.jsdelivr.net/npm/[email protected]/dist/vue-router.min.js',
      'https://cdn.jsdelivr.net/npm/[email protected]/dist/axios.min.js',
      'https://cdn.jsdelivr.net/npm/[email protected]/dist/vuex.min.js',
      'https://cdn.jsdelivr.net/npm/[email protected]/lib/vant.min.js'
    ]
  }
}
module.exports = {
     
  configureWebpack: config => {
     
    // 为生产环境修改配置...
    if (isProduction) {
     
      // externals
      config.externals = externals
    }
  },
  chainWebpack: config => {
     
    /**
     * 添加CDN参数到htmlWebpackPlugin配置中
     */
    config.plugin('html').tap(args => {
     
      if (isProduction) {
     
        args[0].cdn = cdn.build
      } else {
     
        args[0].cdn = cdn.dev
      }
      return args
    })
  }
}

在public/index.html中添加

   <!-- 使用CDN的CSS文件 -->
    <% for (var i in
      htmlWebpackPlugin.options.cdn&&htmlWebpackPlugin.options.cdn.css) {
      %>
      <link href="<%= htmlWebpackPlugin.options.cdn.css[i] %>" rel="preload" as="style" />
      <link href="<%= htmlWebpackPlugin.options.cdn.css[i] %>" rel="stylesheet" />
    <% } %>
     <!-- 使用CDN加速的JS文件,配置在vue.config.js下 -->
    <% for (var i in
      htmlWebpackPlugin.options.cdn&&htmlWebpackPlugin.options.cdn.js) {
      %>
      <script src="<%= htmlWebpackPlugin.options.cdn.js[i] %>"></script>
    <% } %>

对资源文件进行压缩

需要先下载 cnpm i compression-webpack-plugin -D

在vue.config.js中按照下面这种方式进行配置

const CompressionWebpackPlugin = require('compression-webpack-plugin')

module.exports = {
     
  // 根据你的实际情况更改这里
  publicPath,
  assetsDir: 'assets',
  lintOnSave: true,
  configureWebpack: {
     
    plugins:[
      new CompressionWebpackPlugin({
     
        filename: '[path].gz[query]',
        algorithm: 'gzip',
        // test: /\.js$|\.html$|\.json$|\.css/,
        test: /\.js$|\.json$|\.css/,
        threshold: 10240, // 只有大小大于该值的资源会被处理
        minRatio: 0.8, // 只有压缩率小于这个值的资源才会被处理
        // deleteOriginalAssets: true // 删除原文件
      })
    ],
  },
}

图片压缩

需要先下载cnpm i image-webpack-loader --save-dev

进行配置

module.exports = {
     
  // 根据你的实际情况更改这里
  publicPath,
  assetsDir: 'assets',
  lintOnSave: true,
  // image 压缩 定义在chainWebpack中
 chainWebpack: config => {
     
   config.module
      .rule('images')
      .use('image-webpack-loader')
      .loader('image-webpack-loader')
      .options({
     
        bypassOnDebug: true
      })
      .end()}
}

只打包改变的文件

const {
      HashedModuleIdsPlugin } = require('webpack');
configureWebpack: config => {
         
    const plugins = [];
    plugins.push(
        new HashedModuleIdsPlugin()
    )
}

公共代码抽离

从webpack4开始官方移除了commonchunk插件,改用了optimization属性进行更加灵活的配置,这也应该是从V3升级到V4的代码修改过程中最为复杂的一部分

splitChunks: {
     
    chunks: "async”,//默认作用于异步chunk,值为all/initial/async/function(chunk),值为function时第一个参数为遍历所有入口chunk时的chunk模块,chunk._modules为chunk所有依赖的模块,通过chunk的名字和所有依赖模块的resource可以自由配置,会抽取所有满足条件chunk的公有模块,以及模块的所有依赖模块,包括css
    minSize: 30000,  //表示在压缩前的最小模块大小,默认值是30kb
    minChunks: 1,  // 表示被引用次数,默认为1
    maxAsyncRequests: 5,  //所有异步请求不得超过5
    maxInitialRequests: 3,  //初始话并行请求不得超过3
   automaticNameDelimiter:'~',//名称分隔符,默认是~
    name: true,  //打包后的名称,默认是chunk的名字通过分隔符(默认是~)分隔
    cacheGroups: {
      //设置缓存组用来抽取满足不同规则的chunk,下面以生成common为例
       common: {
     
         name: 'common',  //抽取的chunk的名字
         chunks(chunk) {
      //同外层的参数配置,覆盖外层的chunks,以chunk为维度进行抽取
         },
         test(module, chunks) {
       //可以为字符串,正则表达式,函数,以module为维度进行抽取,只要是满足条件的module都会被抽取到该common的chunk中,为函数时第一个参数是遍历到的每一个模块,第二个参数是每一个引用到该模块的chunks数组。自己尝试过程中发现不能提取出css,待进一步验证。
         },
        priority: 10,  //优先级,一个chunk很可能满足多个缓存组,会被抽取到优先级高的缓存组中
       minChunks: 2,  //最少被几个chunk引用
       reuseExistingChunk: true//  如果该chunk中引用了已经被抽取的chunk,直接引用该chunk,不会重复打包代码
       enforce: true  // 如果cacheGroup中没有设置minSize,则据此判断是否使用上层的minSize,true:则使用0false:使用上层minSize
       }
    }
}

第三方模块抽离

页面中有时会引入第三方模块,比如import $ from ‘jquery’;
page1中需要引用,page2中也需要引用,这时候就可以用vendor把jquery抽离出来,如下

// 公共代码抽离
configureWebpack: config => {
     
//....
//优化项配置
config.optimization = {
     
    splitChunks: {
      // 分割代码块
        cacheGroups: {
     
            vendor: {
     //第三方库抽离
                chunks: 'all',
                test: /node_modules/,
                name: 'vendor',
                minChunks: 1,//在分割之前,这个代码块最小应该被引用的次数
                maxInitialRequests: 5,
                minSize: 0,//大于0个字节
                priority: 100//权重
            },
            common: {
       //公用模块抽离
                chunks: 'all',
                test: /[\\/]src[\\/]js[\\/]/,
                name: 'common',
                minChunks: 2,在分割之前,这个代码块最小应该被引用的次数
                maxInitialRequests: 5,
                minSize: 0,//大于0个字节
                priority: 60
            },
            styles: {
      //样式抽离
                name: 'styles',
                test: /\.(sa|sc|c)ss$/,
                chunks: 'all',
                enforce: true
            },
            runtimeChunk: {
     
                name: 'manifest'
            }
        }
    }
}
}

配置打包分析

安装cnpm i webpack-bundie-analyzer -D

const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin

module.exports = {
     
  chainWebpack: config => {
     
    // 打包分析
    if (IS_PROD) {
     
      config.plugin('webpack-report').use(BundleAnalyzerPlugin, [
        {
     
          analyzerMode: 'static'
        }
      ])
    }
  }
}

骨架屏

安装插件cnpm i vue-skeleton-webpack-plugin

在src下新建Skeleton文件夹,其中新建index.js以及index.vue,在其中写入以下内容,其中,骨架屏的index.vue页面样式请自行编辑

index.js

import Vue from 'vue'
import home from './index.vue'
import list from './list.vue'
export default new Vue({
     
  components: {
     
    home,
    list
  },
  template: `
  <div>
   <home id="home" style="display:none"/>
   <list id="list" style="display:none"/>
  </div>
 `
})

index.vue(骨架屏页面) list.vue同理

<template>
  <div class="skeleton-wrapper">
    <header class="skeleton-header"></header>
    <section class="skeleton-block">
      <img src="">
      <img src="">
    </section>
  </div>
</template>
 
<script>
  export default {
     
    name: 'skeleton'
  }
</script>
 
<style scoped>
  .skeleton-header {
     
    height: 40px;
    background: #1976d2;
    padding:0;
    margin: 0;
    width: 100%;
  }
  .skeleton-block {
     
    display: flex;
    flex-direction: column;
    padding-top: 8px;
  }
 
</style>

vue.config.js 配置

//骨架屏渲染
const SkeletonWebpackPlugin = require('vue-skeleton-webpack-plugin')

//path引入
const path = require('path')

//configureWebpack模块中写入内容
// 骨架屏渲染
config.plugins.push(new SkeletonWebpackPlugin({
     
      webpackConfig: {
     
        entry: {
     
          app: path.join(__dirname, './src/Skeleton/index.js'),
        },
      },
      minimize: true,
      quiet: true,
      // 如果不设置那么所有的路由都会共享这个骨架屏组件
      router: {
     
        mode: 'hash',
        // 给对应的路由设置对应的骨架屏组件,skeletonId的值根据组件设置的id
        routes: [
	      {
      path: '/home', skeletonId: 'home' },
	      {
      path: '/list', skeletonId: 'list' },
	    ]
    }))

完整的配置

vue.config.js完整配置

const path = require('path');
const UglifyJsPlugin = require('uglifyjs-webpack-plugin') // 去掉注释
const CompressionWebpackPlugin = require('compression-webpack-plugin'); // 开启压缩
const {
      HashedModuleIdsPlugin } = require('webpack');

function resolve(dir) {
     
    return path.join(__dirname, dir)
}

const isProduction = process.env.NODE_ENV === 'production';

// cdn预加载使用
const externals = {
     
    'vue': 'Vue',
    'vue-router': 'VueRouter',
    'vuex': 'Vuex',
    'axios': 'axios',
    "element-ui": "ELEMENT"
}

const cdn = {
     
    // 开发环境
    dev: {
     
        css: [
            'https://unpkg.com/element-ui/lib/theme-chalk/index.css'
        ],
        js: []
    },
    // 生产环境
    build: {
     
        css: [
            'https://unpkg.com/element-ui/lib/theme-chalk/index.css'
        ],
        js: [
            'https://cdn.jsdelivr.net/npm/[email protected]/dist/vue.min.js',
            'https://cdn.jsdelivr.net/npm/[email protected]/dist/vue-router.min.js',
            'https://cdn.jsdelivr.net/npm/[email protected]/dist/vuex.min.js',
            'https://cdn.jsdelivr.net/npm/[email protected]/dist/axios.min.js',
            'https://unpkg.com/element-ui/lib/index.js'
        ]
    }
}

module.exports = {
     

    lintOnSave: false, // 关闭eslint
    productionSourceMap: false,
    publicPath: './', 
    outputDir: process.env.outputDir, // 生成文件的目录名称
    chainWebpack: config => {
     

        config.resolve.alias
            .set('@', resolve('src'))

        // 压缩图片
        config.module
            .rule('images')
            .test(/\.(png|jpe?g|gif|svg)(\?.*)?$/)
            .use('image-webpack-loader')
            .loader('image-webpack-loader')
            .options({
      bypassOnDebug: true })

        // webpack 会默认给commonChunk打进chunk-vendors,所以需要对webpack的配置进行delete
        config.optimization.delete('splitChunks')

        config.plugin('html').tap(args => {
     
            if (process.env.NODE_ENV === 'production') {
     
                args[0].cdn = cdn.build
            }
            if (process.env.NODE_ENV === 'development') {
     
                args[0].cdn = cdn.dev
            }
            return args
        })

        config
            .plugin('webpack-bundle-analyzer')
            .use(require('webpack-bundle-analyzer').BundleAnalyzerPlugin)
    },

    configureWebpack: config => {
     
        const plugins = [];

        if (isProduction) {
     
            plugins.push(
                new UglifyJsPlugin({
     
                    uglifyOptions: {
     
                        output: {
     
                            comments: false, // 去掉注释
                        },
                        warnings: false,
                        compress: {
     
                            drop_console: true,
                            drop_debugger: false,
                            pure_funcs: ['console.log']//移除console
                        }
                    }
                })
            )
            // 服务器也要相应开启gzip
            plugins.push(
                new CompressionWebpackPlugin({
     
                    algorithm: 'gzip',
                    test: /\.(js|css)$/,// 匹配文件名
                    threshold: 10000, // 对超过10k的数据压缩
                    deleteOriginalAssets: false, // 不删除源文件
                    minRatio: 0.8 // 压缩比
                })
            )

            // 用于根据模块的相对路径生成 hash 作为模块 id, 一般用于生产环境
            plugins.push(
                new HashedModuleIdsPlugin()
            )

            // 开启分离js
            config.optimization = {
     
                runtimeChunk: 'single',
                splitChunks: {
     
                    chunks: 'all',
                    maxInitialRequests: Infinity,
                    minSize: 1000 * 60,
                    cacheGroups: {
     
                        vendor: {
     
                            test: /[\\/]node_modules[\\/]/,
                            name(module) {
     
                                // 排除node_modules 然后吧 @ 替换为空 ,考虑到服务器的兼容
                                const packageName = module.context.match(/[\\/]node_modules[\\/](.*?)([\\/]|$)/)[1]
                                return `npm.${
     packageName.replace('@', '')}`
                            }
                        }
                    }
                }
            };

            // 取消webpack警告的性能提示
            config.performance = {
     
                hints: 'warning',
                //入口起点的最大体积
                maxEntrypointSize: 1000 * 500,
                //生成文件的最大体积
                maxAssetSize: 1000 * 1000,
                //只给出 js 文件的性能提示
                assetFilter: function (assetFilename) {
     
                    return assetFilename.endsWith('.js');
                }
            }

            // 打包时npm包转CDN
            config.externals = externals;
        }

        return {
      plugins }
    },

    pluginOptions: {
     
        // 配置全局less
        'style-resources-loader': {
     
            preProcessor: 'less',
            patterns: [resolve('./src/style/theme.less')]
        }
    },
    devServer: {
     
        open: false, // 自动启动浏览器
        host: '0.0.0.0', // localhost
        port: 6060, // 端口号
        https: false,
        hotOnly: false, // 热更新
        proxy: {
     
            '^/sso': {
     
                target: process.env.VUE_APP_SSO, // 重写路径
                ws: true,   //开启WebSocket
                secure: false,      // 如果是https接口,需要配置这个参数
                changeOrigin: true
            }
        }
    }
}

你可能感兴趣的:(vue)