规则5:压缩组件
压缩组件可以使响应包变小,缩短传输时间。HTTP1.1中,可以通过Accept-Encoding:gzip
(gzip是目前最理想的压缩方法)进行控制(上述已提及)。需要注意的是:图片和pdf不应该压缩,因为它们本来就已经被压缩过了,试图对它们压缩只会浪费CPU资源,还可能增加文件大小。
代理缓存
上述阐述的方式对于浏览器和服务器直接通信会工作的很好,当浏览器通过代理发送过来请求时,情况就复杂了,综述一下具体请查看:Vary、mod_gzip
- 网站用户少或注重带宽开销:
Vary: Accept-Encoding
Web服务器告知代理服务器根据Accept-Encoding来改变缓存的响应(边缘情况太对,尽量不要使用)。
- 网站用户群体大且多变,能过应付带宽开销:
Cache-Control: Private
禁用缓存。
规则6:将样式表放在顶部
将DHTML特征的样式表放在文档顶部Head中首先下载它们能使页面呈现得更快。
无样式内容的闪烁
白屏现象源自浏览器的行为。样式表在页面中的位置并不影响下载时间,但是会影响页面的呈现。
如果样式表仍在加载,构建呈现树就是一种浪费,因为在所有样式表加载并解析完毕之前无需绘制任何东西。否则,在其准备好之前显示内容会遇到FOUC(无样式内容的闪烁 Flash of Unstyled Content)问题。
白屏是浏览器对FOUC问题的补充。浏览器可以延迟呈现,直到所有的样式表都下载完之后,然而,其会导致白屏。反之,浏览器可以逐步呈现,但要承担闪烁的风险。这里没有完美的选择。IE通常会白屏,Firefox等会其他浏览器会闪烁(逐步呈现)。
避免白屏和闪烁:
-
@import url()
会导致组件下载时的无序性,使用Link标签代理会带来性能上的收益; - 如果样式表不是呈现页面所必需的,可以想办法再文档加载完后动态加载;
- 可视化回馈的重要性:(1)让用户知道系统并没有崩溃;(2)告知用户需要等待多久,以便用户可以在漫长等待中做些其他事情;(3)提供一些可以看得东西。
规则7:将脚本放在底部
将脚本放在页面底部,这样可以提高下载的并行速度,同时达到页面逐步呈现。
并行下载
对响应时间影响最大的是页面中组件的数量。HTTP1.1的RFC2616中建议单用户客户端不应该与任何服务器或代理保持超过2个连接,RFC7230中取消了该限制。现代浏览器,一般允许同域6个并发请求。我们可以使用CNAME(DNS别名)将组件分别放到多个主机名中,增加并发下载数。但是增加并发下载数,同时需要取决你的带宽和CPU速度,过多的并行下载反而会降低性能。
脚本阻塞下载
在下载脚本时并行下载实际上是被禁用的—即使用了不同的主机名,浏览器也不会启动其他的下载。之所以做这样的限制有两个原因:(1)脚本可能使用document.write来修改页面内容,因此浏览器会等待,以确保页面能够恰当布局;(2)为了保证脚本能够按照正确的顺序执行。
因此将脚本放到页面顶部不仅会阻塞对其后面内容的呈现,而且还会阻塞后续组件的下载。当然,也可以使用Defferred(延迟)脚本(不包含document.write),浏览器获得这一信息后可继续呈现和下载。
规则8:避免CSS表达式
CSS表达式是动态设置CSS属性的一种强大(并危险)的方式(只针对IE浏览器,其他浏览器不起作用)。在IE11以前的版本,并不支持min-width,通过CSS表达式可以很好的解决该问题。
表达式不只在页面呈现和大小改变时求值,当页面滚动、甚至用户鼠标在页面上拖拽时都要求值。这很可能导致页面死掉,不得不终止进程。
解决表达式重复求值
一次性表达式:可以在表达式执行过程中重写它自身。
事件处理器:可以通过事件处理器达到期望的行为。
注意:没有深入了解底层影响的情况下使用CSS表达式是很危险的!
规则9:减少DNS查找
DNS(Domain Name System,域名系统 )将主机名映射到IP地址上(域名解析)。在解析完成之前,浏览器不能从主机名服务器下载任何东西,而这个过程需要花费一定的时间。其依赖于DNS解析器(ISP提供)、它所承受的请求压力、距离和带宽等。
操作系统具有自身的ISP,同时浏览器也可缓存DNS记录。TTL存活时间决定了域名解析在DNS服务器中存留时间。对于大部分公司都会进行快速故障转移的构建(虚拟IP等),这从一定程度上需要TTL时间不能过长。
- Keep-Alive持久连接,无需DNS查找。
- 减少主机数量(和并行下载有冲突),建议将组件分别放到2到4个主机名下,减少DNS查找和高度并行可以不错的权衡。
规则10:其他
- 压缩CSS和JavaScript;
- 删除重复脚本。这里更多的是指避免重复脚本加载和执行,确保加载过得脚本不被重复加载。
- 避免重定向,如必须重定向,最好使用3xx HTTP状态码,已确保后退按钮可以正常工作;
在URL的结尾必须出现斜线(/)而没有出现 - 使Ajax可缓存。对于一个用户可能每天或者每周进行很多次请求,可以使用Expires头设置缓存,会有带来不错的用户体验。将URL查询字符串携带特征信息(如时间戳)进行重新请求。这里我们携带当前小时的时间戳来达到当前小时内的缓存效果。
总结
上述描述了我们经常使用,也是最基础的前端优化方式或称为最佳实践。作为前端工程师,提高网站性能是我们义不容辞的责任,从而给用户创建更好和更快的页面和体验。
- 减少HTTP请求
- 使用内容发布网络CDN
- 为组件添加长久的Max-Age或Expires头
- 自定义ETage或移除ETag
- 压缩脚本和样式表
- 将JavaScript和CSS放到外部文件中,并确保脚本仅被包含一次
- 使用LINK标签,并将标签放到页面HEAD中
- 将脚本放到页面底部
- 避免CSS表达式
- 通过Keep-Alive和较少的域名较少DNS查找
- 寻找一种避免重定向的方法