网站的速度是关系用户体验的最重要因素之一,谁都会愿意得到快速的反馈,等待是一件让人烦躁的事情。所以性能优化对于网站来说是非常重要的的事情,它带来的好处不用解释。
问题:如果跑车是轿车速度的两倍,是跑车快还是轿车快?
- 比赛1:从起点到终点的比赛:毫无疑问跑车肯定先到达终点,秒杀轿车;
- 比赛2:将 4 个人从起点运到终点:跑车此时就比较尴尬,因为只能带 1 位乘客,而轿车可以一次带 4 位乘客,显然跑车要来回好几趟,而轿车只需要 1 趟,明显轿车的效率更高。
所以性能的优化有两种:一种是快,体现在战术上,通过极致的优化算法达到目的;一种是高效,体现在战略上,通过合理的安排资源达到目的。快就是竞技赛强者胜;而高效是团体赛谋者胜;接下来我们会通过这两个方面对 WEB 前端做一次性能优化之旅。
前端性能优化,首先要搞清楚浏览器页面请求的整个过程,这样我们才能很好的解决问题。
- navigationStart:准备加载新页面;
- unloadEventStart:开始卸载前一个页面;
- unloadEventEnd:页面卸载完成;
- redirectStart:开始重定型;
- redirectEnd:重定向完成;
- fetchStart:开始获取页面;
- domainLookupStart:开始 DNS 解析;
- domainLookupEnd:DNS 解析完成;
- connectStart:开始建立连接;
- connectEnd:建立连接完成;
- secureConnectionStart:开始建立 SSL 连接;
- requestStart:开始请求页面文档;
- responseStart:开始返回页面文档;
- responseEnd:返回页面文档完成;
- domLoading:当前页面开始解析;(domReady 白屏时间)
- domInteractive:页面解析完成,还没有开始加载资源(样式、脚本、图片);
- domContentLoadedEventStart:页面解析完成,开始加载资源(样式、脚本、图片),开始运行脚本(同步的脚本);
- domContentLoadedEventEnd:脚本运行完成(同步的脚本下载完成,首屏时间)
- domComplete:资源(图片等)加载完成
- loadEventStart:触发 load 事件的开始时间(onload)
- loadEventEnd:触发 load 事件的结束时间
- 白屏时间(domLoading):
- HTML
- 首屏时间(domContentLoadedEventEnd):
- 样式
- 脚本(核心脚本)
- 页面加载完成(domComplete):
- 图片
- 样式
- 所有资源加载完成(loadEventStart):
- 样式
- 脚本(最底部脚本,通常是统计脚本)
- 图片
- iFrame(我们没有)
前人已经给我们留下了很多 WEB 性能优化的准则,我们一一列举出来(来自雅虎 YSlow 团队对 Web 性能的总结):
- [术]减重传输
- 使用小 favicon.ico 文件
- 减少 Cookie 大小
- 静态资源文件使用无 Cookie 的域
- 精简 JavaScript 和 CSS
- 移除重复的脚本
- 使用 Gzip 压缩
- [略]打包传输
- 尽可能减少 HTTP 请求次数,合并 JavaScript 和 CSS
- IconFont
- [略]合理的加载顺序
- 在 HTML 文件顶部放置样式表
- 在 HTML 文件底部放置 JavaScript 脚本
- 图片按需加载
- [术]利用缓存
- 使用外部 JavaScript 和 CSS 外部文件
- 加入 Expires 或 Cache-Control Header
- 配置 ETag
- 缓存 AJAX
- 使用 GET 完成 AJAX 请求
- [术]网络层
- 使用 CDN
我们的 WEB 框架本身就完成了一些性能上的优化:
- 构建工具
- 压缩合并静态文件
- 生成小的 favicon.ico 文件
- 合理的动态引用 JavaScript
- 样式在前面,JavaScript 在后面
- 自动拆分静态资源成为外部资源
- Node & Nginx
- Gzip
- Expires 和 Cache-Control Header
- ETag
为了看清楚慢到底在哪儿,通过一个浏览器来调试这种方式是不科学的,我们需要更多用户的数据分析数据,我们在网站中植入了性能收集的代码(听云),每个用户的性能都可以统计到,这样才能找到真正慢在哪儿。
我们分析了所有慢的网页,发现了网站慢的普片原因:图片加载太慢!
- 站内图片
- 保证不影响视觉的情况下,压缩大小
- 小图片 Base64
- icon 采用 iconfont 将图标文件一次性输出
- 站外图片
- 采用按需加载
- 分享二维码
- 组件中的图片
- 整个组件按需加载
- 站外脚本
- ABTesting
- 快速显示 Dom
优化完了我们不能说就结束了,我们需要看最终结果是不是好的,是否是真的达到了我们的目的,通过数据可以验证我们的方案是否成功,如果不成功到底错在哪里,我们需要如何改进。