详解Vue SPA项目优化小记


Posted in Javascript onJuly 03, 2018

概述

之前做了一个React项目和Vue的项目,在做完后不加任何优化的情况下,这2个项目的首屏加载平均时间居然达到了20+s,好一点能进10s,差一点快30s,完全不能忍,优化势在必行,本文章记录下Vue项目的优化过程,React项目的优化后续补上。

详解Vue SPA项目优化小记

上图是Vue项目的首页,整个项目由vue-cli搭建,主要分为4个模块,见页面左侧导航栏,总体代码量不算太大,组件的话总共50个左右吧,项目结构见下图,不算特别大的项目,但是首屏加载时间居然这么慢。

详解Vue SPA项目优化小记

首先得确定到底是哪里导致了首屏渲染如此之慢?打开Chrome Network面板,勾上Disable cache选项,刷新页面观察资源加载情况,发现罪魁祸首就是webpack打包生成的app.js和vendor.js,其中vendor.js大小达到了1.2M,下载时间超过20秒,app.js也快到1M,而manifest.js不是很大。vendor.js主要是把node_modules里所用到的modules都合并成一个js了,所以比较大.而我们也希望将业务代码和第三方引用分开打包。 manifest.js 包含webpack的runtime代码和module manifest代码,作用是防止修改了代码但是没有修改第三方库文件导致第三方库文件也打包的问题。

网上一番搜索,发现优化点主要在如下几个方面:

  1. 开启gzip压缩功能
  2. 引入CDN
  3. 路由懒加载
  4. 某些第三方组件按需加载而不是全部加载
  5. 较小的图片资源用base64嵌入src中,减少http请求

具体优化

gzip压缩

gzip是GNUzip的缩写,最早用于UNIX系统的文件压缩。HTTP协议上的gzip编码是一种用来改进web应用程序性能的技术,web服务器和客户端(浏览器)必须共同支持gzip。目前主流的浏览器,Chrome,firefox,IE等都支持该协议。常见的服务器如Apache,Nginx,IIS同样支持,gzip压缩效率非常高,通常可以达到70%的压缩率,也就是说,如果你的网页有30K,压缩之后就变成了9K左右

我后端是用的express,开启gzip非常简单,首先npm install compression安装中间件,然后在app.js里添加use使用即可:

var compression = require('compression');
var app = express();
app.use(compression())

重启服务,观察网络面板里面的response header,如果看到如下红圈里的字段则表明gzip开启成功

详解Vue SPA项目优化小记

引入CDN

CDN(内容分发网络),是一种公共服务,他本身有很多台位于不同地域、接入不同运营商的服务器,而所谓的使用CDN实质上就是让CDN作为网站的门面,用户访问到的是CDN服务器,而不是直接访问到网站。由于CDN内部对TCP的优化、对静态资源的缓存、预取,加上用户访问CDN时,会被智能地分配到 最近 的节点,降低大量延迟,让访问速度可以得到很大提升

一个原则是尽量将比较大的第三方库放到cdn上去以减少请求时间,在我的项目中,我将vue,vuex,vue-router,echarts都放到了cdn上,具体操作是打开BootCDN 然后搜索关键字并copy链接粘贴到index.html的body闭合标签前,如下图

详解Vue SPA项目优化小记

注意选取min.js(体积最小),然后在webpack.base.conf.js里设置externals选项,目的是不打包这些选项,由于index.html中script的引入,比如vue就会有一个全局变量Vue存在,因此这里external的value就是Vue

externals: {
 'vue': 'Vue',
 'vuex': 'Vuex',
 'vue-router': 'VueRouter',
 'echarts': 'echarts'
 },

cdn使用后优势是巨大的,观察network面板,时间几乎都在50ms以下

详解Vue SPA项目优化小记

路由懒加载

路由懒加载也叫延迟加载,即在需要的时候进行加载,随用随载。像vue这种单页面应用,如果没有应用懒加载,运用webpack打包后的文件将会异常的大,造成进入首页时,需要加载的内容过多,时间过长,会出现长时间的白屏,即使做了loading也是不利于用户体验,而运用懒加载则可以将页面进行划分,需要的时候加载页面,可以有效的分担首页所承担的加载压力,减少首页加载用时。

Vue官网的示例如下,采用异步组件和webpack的code-splitting结合

详解Vue SPA项目优化小记

因此在项目中,进入router的index.js中,将原来的 import Comp from '@/component/xxx' 改为如下,vue-router的配置项还是保持不变

//异步组件,路由懒加载
const BookMark = resolve => require(['@/components/BookMark'],resolve);
const HotBookMark = resolve => require(['@/components/HotBookMark'],resolve);
const ImportBookMark = resolve => require(['@/components/ImportBookMark'],resolve);
const Default = resolve => require(['@/components/Default'],resolve);

打包后查看js文件夹下的文件,会多出上述文件,每个进行懒加载的组件都会生成一个js,如下图红框内

详解Vue SPA项目优化小记

第三方按需加载

比如本项目里面使用的echarts,只用到了一个柱状图组件,其余的都没有用到,但是这样import后打包时却会把整个echarts都打入包内,造成空间浪费

import echarts from 'echarts'

因此只需要import用到的组件即可,如下,这样就可以减少很多不必要的体积

import echarts from 'echarts/lib/echars'
import 'echarts/lib/chart/bar'
import 'echarts/lib/component/legend'
import 'echarts/lib/component/title'

图片转base64

小图片可以转为base64字符串然后嵌入img的src中,节省http请求数量,webpack中用url-loader处理,limit控制了图片转base64的阈值,小于该值就转base64

{
 test: /\.(png|jpe?g|gif|svg)(\?.*)?$/,
 loader: 'url-loader',
 options: {
  limit: 10000,
  name: utils.assetsPath('img/[name].[hash:7].[ext]')
 }
 },

优化后性能

经过上述优化后,首页打开时间迅速下降,DomContentLoaded用时不到1s,load完全加载用时不到4s,耗时较多的是几张背景大图,本来体积就大,后续考虑放到cdn上

详解Vue SPA项目优化小记

这是最初的优化策略,后续继续会进行深度优化。

ps:webpack-bundle-analyzer是神器,能够有效分析出包占用的体积情况~下图是最终优化后的包组成结构图,最初打包的结构图比下图大很多,主要多了echarts,vue,vue-router,vuex模块

详解Vue SPA项目优化小记

优化后的整个包gzip后大小喜人~

详解Vue SPA项目优化小记

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jquery实现的超出屏幕时把固定层变为定位层的代码
Feb 23 Javascript
struts2+jquery+json实现异步加载数据(自写)
Jun 24 Javascript
简述JavaScript的正则表达式中test()方法的使用
Jun 16 Javascript
AngularJS的一些基本样式初窥
Jul 27 Javascript
异步JavaScript编程中的Promise使用方法
Jul 28 Javascript
jQuery实现点击后高亮背景固定显示的菜单效果【附demo源码下载】
Sep 21 Javascript
使用BootStrapValidator完成前端输入验证
Sep 28 Javascript
js实现PC端和移动端刮卡效果
Mar 27 Javascript
微信小程序实现图片上传功能实例(前端+PHP后端)
Jan 10 Javascript
浅谈React的最大亮点之虚拟DOM
May 29 Javascript
微信小程序实现图片翻转效果的实例代码
Sep 20 Javascript
JS Array.from()将伪数组转换成数组的方法示例
Mar 23 Javascript
jQuery实现表单动态添加与删除数据操作示例
Jul 03 #jQuery
JS实现显示当前日期的实例代码
Jul 03 #Javascript
jQuery实现获取form表单内容及绑定数据到form表单操作分析
Jul 03 #jQuery
vue 设置路由的登录权限的方法
Jul 03 #Javascript
jQuery阻止事件冒泡实例分析
Jul 03 #jQuery
详解VUE中常用的几种import(模块、文件)引入方式
Jul 03 #Javascript
Vue props用法详解(小结)
Jul 03 #Javascript
You might like
列举PHP的Yii 2框架的开发优势
2015/07/03 PHP
两种php给图片加水印的实现代码
2020/04/18 PHP
ThinkPHP函数详解之M方法和R方法
2015/09/10 PHP
PHP带节点操作的无限分类实现方法详解
2016/11/09 PHP
PHP项目多语言配置平台实现过程解析
2020/05/18 PHP
javascript fullscreen全屏实现代码
2009/04/09 Javascript
优化javascript的执行速度
2010/01/23 Javascript
z-blog SyntaxHighlighter 长代码无法换行解决办法(jquery)
2014/11/16 Javascript
使用javascript实现监控视频播放并打印日志
2015/01/05 Javascript
jQuery简单实现图片预加载
2015/04/20 Javascript
JS+CSS实现自动切换的网页滑动门菜单效果代码
2015/09/14 Javascript
使用开源工具制作网页验证码的方法
2016/10/17 Javascript
Json按某个键的值进行排序
2016/12/22 Javascript
vue语法之拼接字符串的示例代码
2017/10/25 Javascript
JS实现DOM删除节点操作示例
2018/04/04 Javascript
如何解决webpack-dev-server代理常切换问题
2019/01/09 Javascript
JS基于对象的链表实现与使用方法示例
2019/01/31 Javascript
ES6基础之默认参数值
2019/02/21 Javascript
使用 Vue cli 3.0 构建自定义组件库的方法
2019/04/30 Javascript
Flutter 超实用简单菜单弹出框 PopupMenuButton功能
2019/08/06 Javascript
JavaScript缺少insertAfter解决方案
2020/07/03 Javascript
vue中提示$index is not defined错误的解决方式
2020/09/02 Javascript
python使用pygame实现笑脸乒乓球弹珠球游戏
2019/11/25 Python
ansible动态Inventory主机清单配置遇到的坑
2020/01/19 Python
python文件编写好后如何实践
2020/07/07 Python
Python如何测试stdout输出
2020/08/10 Python
利用Python函数实现一个万历表完整示例
2021/01/23 Python
Canvas环形饼图与手势控制的实现代码
2019/11/08 HTML / CSS
车间班组长的职责
2013/12/13 职场文书
初一地理教学反思
2014/01/16 职场文书
秋游活动策划方案
2014/02/16 职场文书
团日活动总结
2014/04/28 职场文书
监察建议书格式
2014/05/19 职场文书
考研英语复习计划
2015/01/19 职场文书
房屋买卖定金协议书
2016/03/21 职场文书
「回转企鹅罐」10周年纪念展「輪るピングドラム展」海报公开
2022/03/22 日漫