最近半年在维护公司的一个管理后台项目,搭建之初的技术栈比较混乱,构建方案采用了Gulp中调用Webpack的方式,Gulp负责处理.html文件,Webpack负责加载.vue.js等。而在这一套构建方案中,主要有这些问题:

  1. 没有实现JS压缩、CSS兼容等功能。
  2. 在开发模式下,保存代码,项目会进行完全的重新打包,持续构建速度不仅缓慢,还会产生缓存的现象(构建完成后刷新页面改动不生效)。
  3. 由于目前的方案没有使用http-proxy-middleware这样的请求代理模块,导致项目在本地开发时还要部署后端服务,对新接手的开发者不友好,而且经常由于沟通不及时产生测试环境与本地环境的代码同步问题。

因此,在熟悉这个项目之后,打算对其构建方案进行升级,主要为了解决上述的问题。

1. 原有构建方案描述

原有构建速度

  • npm run build:打包约50s
  • npm run dev:开启开发模式约50s,保存自动重新编译需约6s,编译完成后需要刷新才能看到效果,偶尔因缓存问题需要再次自动重新编译才能看到效果

原有构建结果

  • ./build/development:存放渲染后的js文件
  • ./build/html:存放渲染后的html文件
  • ./build/rev:保存各个入口文件hash值的json文件

打包代码解析

/*** 使用gulp-clean插件删除build目录下的文件*/
gulp.task('clean', function () {if (!stopClean) {return gulp.src('build/' + directory, { read: false }).pipe(clean())}
})
/*** 使用webpack打包vue与js文件,在clean之后进行*/
gulp.task('webpack', ['clean'], function (callback) {deCompiler.run(function (err, stats) {if (err) throw new gutil.PluginError('webpack', err)gutil.log('[webpack]', stats.toString({}))callback()})
})
/*** 使用gulp-uglify插件对js文件进行丑化,在webpack之后进行*/
gulp.task('minify', ['webpack'], function () {if (environment) {return} else {return gulp.src('build/' + directory + '/*.js').pipe(uglify())}
})
/*** 使用gulp-rev插件为打包后的文件增加hash,在minify之后运行* * gulp-rev会做什么:* 根据静态资源内容,生成md5签名,打包出来的文件名会加上md5签名,同时生成一个json用来保存文件名路径对应关系。* 替换html里静态资源的路径为带有md5值的文件路径,这样html才能找到资源路径。* 有些人可能会做:静态服务器配置静态资源的过期时间为永不过期。* 达到什么效果:* 静态资源只需请求一次,永久缓存,不会发送协商请求304* 版本更新只会更新修改的静态资源内容* 不删除旧版本的静态资源,版本回滚的时候只需要更新html,同样不会增加http请求次数*/
gulp.task('hashJS', ['minify'], function () {var dest = gulp.src(['一串入口文件...']).pipe(rev()) // 设置文件的hash key.pipe(gulp.dest('build/' + directory)) // 将经过管道处理的文件写出到目录.pipe(rev.manifest({})) // 生成映射hash key的json.pipe(gulp.dest('build/rev')) // 将经过管道处理的文件写出到目录!environment && gulp.src(['一串入口文件...']).pipe(clean())return dest
})
/*** 使用gulp-rev-replace插件为html中引用的js和css替换新的hash* 使用gulp-livereload插件在所有文件重新打包完成后局部更新页面*/
gulp.task('revReplace', ['hashJS'], function () {return gulp.src(['html/*.html']).pipe(revReplace({ ... })) // 给html中的js引用提供新的hash.pipe(gulp.dest('build/html')) // 输出文件.pipe(livereload()) // 局部更新页面
})
/*** 使用gulp.watch,当应用程序目录下有任何文件发生改变,则重新执行一遍打包命令* gulp.watch:监视文件,并且可以在文件发生改动时候做一些事情。*/
gulp.task('watch', ['revReplace'], function () {stopClean = truelivereload.listen()gulp.watch('app/**/*', ['clean', 'webpack', 'minify', 'hashJS', 'revReplace'])
})
/*** 输出dev和build的工作流*/
gulp.task('default', ['clean', 'webpack', 'minify', 'hashJS', 'revReplace', 'watch']) // dev
gulp.task('build', ['clean', 'webpack', 'minify', 'hashJS', 'revReplace']) // build
/*** webpack配置*/
var devCompiler = webpack({entry: {... // 一众入口文件vendor: ['vue', 'vue-router', 'lodash', 'echarts'] // 公共模块},output: {path: ..., // 所有输出文件的目标路径publicPath: ..., // 输出解析文件的目录filename: ..., // 输出文件chunkFilename: ... // 通过异步请求的文件},// 排除以下内容打包到 bundle,减小文件大小external: {jquery: 'jQuery',dialog: 'dialog'},plugins: [/*** 通过将公共模块拆出来,最终合成的文件能够在最开始的时候加载一次,便存到缓存中供后续使用。* 这个带来页面速度上的提升,因为浏览器会迅速将公共的代码从缓存中取出来,而不是每次访问一个新页面时,再去加载一个更大的文件。*/new webpack.optimize.CommonsChunkPlugin({name: ['vendor']}),/*** DefinePlugin 允许创建一个在编译时可以配置的全局常量。这可能会对开发模式和生产模式的构建允许不同的行为非常有用。*/new webpack.DefinePlugin({__VERSION__: new Date().getTime()})],resolve: {root: __dirname,extensions: ['', '.js', '.vue', '.json'], // 解析组件的文件后缀白名单alias: { ... } // 配置路径别名},module: {// 各个文件的loadersloaders: [{ test: /\.vue$/, loader: 'vue-loader' },{ test: /\.css$/, loader: 'style-loader!css-loader' },{ test: /\.jsx$/, loader: 'babel-loader', include: [path.join(__dirname, 'app')], exclude: /core/ },{ test: /\.json$/, loader: 'json' }]},vue: {loaders: {js: 'babel-loader'}}
})
复制代码

2. 将Gulp的功能移到Webpack1上执行

使用html-webpack-plugin插件构建项目的主.html文件

module.exports = {plugins: [new HtmlWebpackPlugin({filename: '...', // 输出的路径template: '...', // 提取源html的路径chunks: ['...'], // 需要导入的模块inject: true // 是否附加到body底部})]
}
复制代码

使用webpack.optimize.UglifyJsPlugin插件进行JS压缩

module.exports = {plugins: [new webpack.optimize.UglifyJsPlugin({compress: { warnings: false }})]
}
复制代码

使用webpack-dev-server模块,提供node搭建的开发环境

module.exports = {devServer: {clientLogLevel: 'warning', // 输出日志的级别,配置为警告级别以上才输出inline: true, // 启动 live reloadhot: true, // 允许启用热重载compress: true, // 对所有静态资源进行gzip压缩open: true, // 默认在启动本地服务时打开浏览器quiet: true, // 禁止输出繁杂的构建日志host: ..., // 服务启动的域名port: ..., // 服务启动的端口proxy: { ... }, // http代理配置/*** 这个配置常用于解决spa应用h5路由模式下将所有404路由匹配回index.html的问题* 由于生产环境为主页匹配了一个比较简单的别名,因此开发环境也照搬后端服务的配置*/historyApiFallback: {rewrites: [{ from: '/^\/admin/', to: '...' }]}}
}
复制代码

踩坑

  1. webpack-dev-server@3.2.1 requires a peer of webpack^@4.0.0 but none is installed.:这两个模块版本不兼容,回退到webpack-dev-server@2成功运行。
  2. Cannot resolve module 'fsevents' ...:将全局的webpack调用改为直接从node_modules/webpack下直接调用,解决了问题,node node_modules/webpack/bin/webpack.js --config webpack.config.js
  3. Cannot resolve module 'fs' ...:配置config.node.fs = 'empty',为Webpack提供node原生模块,使其能加载到这个对象。
  4. 热重载只对.js.css.vue中的<style>内样式生效,对.vue文件中的html模板及js内容都不生效,会打印“模块代码已发生改变并重新编译,但热重载不生效,可能会启用全局刷新的策略”之类的信息,暂时没有解决,初步判断是低版本的vue-hot-reload-api对这些部分的处理有问题,有大神了解原理可以在评论区科普一哈=.=。

3. 从Webpack1升级到Webpack3

由于Webpack2Webpack3几乎完全兼容,只是涉及到一些增量的功能,因此选择直接从Webpack1迁移到Webapck3,先在项目中安装Webpack3,然后根据Webpack2文档中《从Webpack1迁移》的章节,对配置项进行更改,参考的文档戳这个:www.html.cn/doc/webpack…

这次升级没有遇到什么问题,根据文档配置稍作更改就跑通了。梳理一下目前为止实现的功能:

  1. 新的Webpack构建代码已经实现了原有的所有功能,下面列举新增的功能。
  2. 使用webpack-dev-server作为开发服务器,实现了保存时live reload的功能。
  3. 使用http-proxy-middleware插件,将请求直接代理到测试服,让开发环境脱离了本地部署的后端服务,大大降低了开发环境部署的时间成本。
  4. 新增friendly-errors-webpack-plugin,输出友好的构建日志,打印几个重要模块的开发环境地址,配置方面完全参考了vue-cli@2的默认配置。
  5. 新增postcss-loader,对css添加兼容处理,配置方面完全参考了vue-cli@2的默认配置。
  6. 使用webpack.optimize.UglifyJsPlugin压缩js代码。

尝试进行构建,输出构建时间记录:

  • npm run build:约135s
  • npm run dev:初次构建约58s,持续构建约30s

项目构建时间过长(第一次打包把自己吓了一跳...),只能继续寻求构建速度上的优化

4. 在Webpack3下进行构建速度的优化

使用webpack-jarvis监测构建性能

webpack-jarvis是一个图形化的webpack性能监测工具,它配置简便,对构建过程的时间占比、构建结果的详细记录都有具体的输出

// 经过简单的配置就可以在本地3001端口输出构建结果记录
const Jarvis = require('webpack-jarvis')
module.exports = {plugins: [new Jarvis({watchOnly: false,port: 3001})]
}
复制代码

使用happypack

先根据网上搜到的文章,做一些简单的优化,如使用happypack,这个模块通过多进程模型,来加速代码构建,但是使用之后貌似没有太明显的结果,构建时间大概减少了几秒吧...暂时还不太懂这个模块对优化什么场景的效果比较明显,之前有看到一篇讲解happypack原理的文章,但还没细看,有兴趣小伙伴可以研究一下,要是能在评论里简洁明了的给渣渣楼主解释一下就更好了TUT:taobaofed.org/blog/2016/1…

const HappyPack = require('happypack')
const os = require('os')
const happyThreadPool = HappyPack.ThreadPool({ size: os.cpus().length })
module.exports = {plugins: [new HappyPack({// happypack的id,在调用时需要声明,若需要编译其他类型的文件需要再声明一个happypackid: 'js',// cacheDirectory:设置后,将尽量在babel编译时使用缓存加载器的结果,避免重新走一遍babel的高昂代价use: [{ loader: 'babel-loader', cacheDirectory: true }],// 根据cpu的核心数判断需要拆分多少个进程池threadPool: happyThreadPool,// 是否输出编译过程的日志verbose: true})]
}
复制代码

做完这一步后,输出构建时间记录:

  • npm run build:约130s
  • npm run dev:初次构建约60s,持续构建约30s

devtool配置为cheap-module-eval-source-map

devtool选项启用cheap-module-eval-source-map模式:vue-cli@2默认配置为这种模式,cheap代表在输出source-map时省略列信息;module表示在编译过程中启用如babel-loader这样的预编译器,使得调试时可以直接看到未经编译的源代码;eval表示启用eval模式编译,该模式直接使用eval函数执行编译后模块的字符串,减少了将字符串转化为可执行的代码文件这个步骤,加快了项目开发中重建的速度;source-map表示输出源代码的映射表,使得开发时可以直接把错误定位到源代码,提高开发效率。

做完这一步后,效果并不明显=.=(相比原来的source-map),大概减少了几秒,输出构建时间记录:

  • npm run build:约130s
  • npm run dev:初次构建约58s,持续构建约30s

使用html-webpack-plugin-for-multihtml提升多入口项目重建速度

重建一次竟然需要30s!各种搜索找到了html-webpack-plugin的一条issue,发现html-webpack-plugin@2在构建多入口应用时速度确实有明显变慢的情况,原因是没有成功的对构建内容进行缓存,使每次重建都重新编译所有代码。作者给出的解决方案是使用这个模块的一个分支项目(是由作者本人fork原项目并针对这个问题进行修复的项目)html-webpack-plugin-for-multihtml,用法与html-webpack-plugin完全相同,使用之后重建仅需1s左右。

做完这一步后,输出构建时间记录:

  • npm run build:约130s
  • npm run dev:初次构建约58s,持续构建约1s

使用webpack.DllPlugin提取公共模块

在输出结果中找到了不少较大的依赖包,如Vue的核心库、lodashecharts等等,还有一些不希望被打包的静态资源,想办法避免每次都编译这些内容,提升编译速度,所以找到了这个插件。

webpack.DllPlugin这个插件是来源于Windows系统的.dll文件(动态链接库)的用法:首先通过DllPlugin模块构建出一个包含公共模块的包和一个映射表,再通过DllReferencePlugin模块通过映射表给每个模块关联对应的依赖,这样可以对这些公共模块进行预先打包,以后构建的时候就不需要处理这些模块,减少打包时间。

// webpack.dll.conf.js
const webpack = require('webpack')
module.exports = {entry: {vendor: [...]},output: {path: resolve('build/development'),filename: '[name].dll.js',library: '[name]_library'},plugins: [new webpack.optimize.UglifyJsPlugin(),new webpack.DllPlugin({path: resolve('build/development/[name]-manifest.json'), // 生成manifest文件输出的位置和文件名称name: '[name]-library', // 与output.library是一样的,对应manifest.json文件中name字段的值,防止全局变量冲突context: __dirname})]
}
// webpack.base.conf.js
const webpack = require('webpack')
module.exports = {plugins: [new webpack.DllReferencePlugin({context: __dirname,manifest: require('../build/development/vendor-manifest.json') // 让webpack从映射表获取使用的依赖})]
}
复制代码

打包出来之后还需要在html文件中引入公共库vendor.dll.js文件

<html><head></head><body><div id="app"></div><script src="/build/development/vendor.dll.js"></script><!-- 其他JS应该注入到dll的后面,确保能够引用到公共库的内容 --></body>
</html>
复制代码

做完这一步后,输出构建时间记录,发现构建效率有了明显的提高:

  • npm run dll:约25s
  • npm run build:约70s
  • npm run dev:初次构建约55s,持续构建约1s

5. 后记

优化到这里就差不多结束,这次的优化为旧项目提供了新一代spa项目应有的一些功能,搭建了更现代的本地开发环境。由于本文篇幅有点太长,完整的配置就丢在另一篇文章里。

6. Q&A

Q: 为什么不直接升级到Webpack4

A: Webpack4只支持vue-loader@15以上版本,而这个版本已经无法解析Vue1的文件。

作者信息

作者其他文章

  • 【JavaScript】不常用知识点复(yù)习(一)
  • 【面试】社招中级前端笔试面试题总结-答案及拓展
  • 【网络】什么是跨域,为什么浏览器会禁止跨域,及其引起的发散性学习

【Vue】Vue1.0+Webpack1+Gulp项目升级构建方案的踩坑路相关推荐

  1. vue+django2.0.2-rest-framework 生鲜项目(八)

    vue+django2.0.2-rest-framework 生鲜项目 一.支付宝沙河环境配置 线上正式: 进入蚂蚁金服开放平台(https://open.alipay.com/platform/ho ...

  2. vue+django2.0.2-rest-framework 生鲜项目(三)

    vue+django2.0.2-rest-framework 生鲜项目 一.xadmin后台管理系统配置 不使用Django自带admin后台管理,使用xadmin后台管理,类似CRM项目中的king ...

  3. 微信vue路由跳转兼容_Vue微信公众号开发踩坑记录

    需求 微信授权登录(基于公众号的登录方案) 接入JS-SDK实现图片上传,分享等功能 现状及难点 采用的Vue框架,前后端分离模式(vue工程仅作为客户端),用户通过域名访问的是客户端,但是微信授权中 ...

  4. 记一次成功把Vue2后台项目改造成Vite2的踩坑经历

    文章目录 前言 一.项目背景 1.1.为什么要选择Vite 二.迁移前的准备 2.1.补全.vue后缀 2.2.移动public/index.html的位置 2.2.1.通过vite-plugin-h ...

  5. Vue项目对接微信公众号踩坑日记

    之前做项目都是pc端的,还是第一次做移动端项目,而且上来就要接入app 和微信公众号两个平台,最终查阅多方文档,耗费几周时间还是完成了项目,这篇文章也算是记录一下自己的完成思路以及一些想法,希望能帮到 ...

  6. vue项目中更换tinymce版本踩坑

    项目需求: vue项目中实现多图片批量上传功能 问题: tinymce富文本编辑器的多图片批量上传插件 支持版本:5.0.4+ 项目中现有的富文本编辑器版本:4.9.4 为实现这一功能选择更换tiny ...

  7. Swift原生项目中集成RN的踩坑笔记

    学习Reate Native的踩坑之路 搭建环境 官方环境搭建地址.官方原生集成地址 本人环境:mac10.15.4.Xcode11.4.brew:2.2.16.Pods:1.9.1.npm:6.14 ...

  8. lgg8各个版本_LG G8 展示机 升级安卓10 防踩坑指南

    先来对比下展示机版本信息是不是大体相同再下手 关于手机中,无s/n,无IMEI 安卓9软件版本为G820UM10C,硬件版本1.0 被阉割功能:5G频段wifi,nfc,移动网络(无基带),高分屏,r ...

  9. ASP.NET Core 2.0 Web API项目升级到ASP.NET Core 3.0概要笔记

    本文结构 先决条件 升级目标框架(Target Framework)的版本 过时的IHostingEnvironment与IApplicationLifetime对象 Endpoint Routing ...

最新文章

  1. 关于ceph源码 backtrace 打印函数调用栈
  2. spring mvc 篇
  3. InstallShield2013 error 6109
  4. optee的栈指针和栈内存的介绍
  5. 编码设置导致了eclipse/myeclipse代码无法保存
  6. 五、性能监视(2)Windows性能日志
  7. 堆结构导致数据文件不能收缩
  8. 【实践】美团外卖广告智能算力的探索与实践
  9. android 梯形按钮_PLC编程入门梯形图实例讲解
  10. 10人勾结苹果外包公司员工窃个人信息 涉案900万
  11. MySQL并行复制的深入浅出
  12. tjh_pipeline_tools Maya工具盒 1.2.1 下载及教程 动画影视团队工具共享系统
  13. 如何打开KML/KMZ文件
  14. powerbi导入地图_在Microsoft Power BI中创建地图的10种方法
  15. 利用setInterval()方法实现在页面上显示实时时间
  16. 三角形外心坐标的计算公式
  17. 一周信创舆情观察(6.8~6.14)
  18. 弱电人要学习的网络安全基础知识
  19. AD怎么输入坐标_双心软件回头曲线坐标计算?
  20. C++_OpenCV分离RGB通道

热门文章

  1. 【quickhybrid】架构一个Hybrid框架
  2. Oracle truncate、 delete、 drop区别
  3. rundeck入门-初步操作
  4. SCDPM2012功能测试(4)—配置通知
  5. Repeater控件的使用
  6. Microsoft Teams Voice语音落地系列-5 实战: Sonus语音网关配置
  7. 「镁客早报」人类首次在太空3D打印生物器官;中国学者研制出高性能低成本的电解“水制氢”催化剂...
  8. ServerSocket01
  9. PHP版本的Graphviz样例之集群流程图
  10. 线程调度四(setDaemon方法的使用)