作为前端应该知道的7类35条前端优化规则,下面是雅虎前端优化规则(Yslow)的翻译。
原文:Best Practices for Speeding Up Your Web Site
目录
1. Content
1. Make Fewer HTTP requests
Minimize HTTP Requests
减少/最小化 http 请求数。
到终端用户的响应时间80%花在前端:大部分用于下载组件(js/css/image/flash等等)。减少组件数就是减少渲染页面所需的http请求数。这是更快页面的关键。
减少组件数的一个方法就是简化页面设计。保持富内容的页面且能减少http请求,有以下几个技术:
Combined files
。合并文件,如合并js,合并css都能减少请求数。如果页面间脚本和样式差异很大,合并会更具挑战性。CSS Sprites
。雪碧图可以合并多个背景图片,通过background-image 和 background-position 来显示不同部分。Image maps
。合并多个图片到一个图片,一般用于如导航条。由于定义坐标的枯燥和易错,一般_不推荐_。Inline images
。使用data:url scheme来內连图片。
减少请求数是为第一次访问页面的用户提高性能的最重要的指导。
2. Reduce DNS Lookups——[ 减少DNS查询 ]
就像电话簿,你在浏览器地址栏输入网址,通过DNS查询得到网站真实IP。
DNS查询
被缓存来提高性能。这种缓存可能发生在特定的缓存服务器(ISP/local area network维护),或者用户的计算机。DNS信息留存在操作系统DNS缓存中(在windows中就是 DNS Client Serve )。大多浏览器有自己的缓存,独立于操作系统缓存。只要浏览器在自己的缓存里有某条DNS记录,它就不会向操作系统发DNS解析请求。
IE默认缓存DNS记录30分钟,FireFox默认缓存1分钟。
当客户端的DNS缓存是空的,DNS查找次数等于页面中的唯一域名数。
减少DNS请求数可能会减少并行下载数。避免DNS查找减少响应时间,但减少并行下载数可能会增加响应时间。指导原则是组件可以分散在至少2个但不多于4个的不同域名。这是两者的妥协。
3. Avoid Redirects ——[ 避免跳转 ]
跳转用301
或302
状态码来达成。一个301
响应http头的例子:
浏览器自动跳转到Location
指定的路径。跳转所需的所有信息都在http头,所以http主体一般是空的。301 302
响应一般不会被缓存,除非有额外的头部信息,比如Expires或Cache-Control指定要缓存。meta
刷新标签或 JavaScript 也可以跳转,但如果真要跳转,3xx
跳转更好,主要是保证返回键可用。
跳转显然拖慢响应速度。在跳转的页面被获取前浏览器没什么能渲染,没什么组件能下载。
最浪费的跳转之一发生在url尾部slash(/)缺失。比如http://astrology.yahoo.com/astrology
会301跳转到http://astrology.yahoo.com/astrology/
。这可以被Apache
等服务器修复,用Alias
,mod_rewrite
等等。
4. Make Ajax Cacheable ——[ Ajax缓存 ]
使用ajax
的好处是可以向用户提供很快的反馈,因为它是向后台异步请求数据。但是,这些异步请求不保证用户等待的时间——异步不意味着瞬时。
提高ajax性能的最重要的方法是让响应被缓存,即在Add an Expires or a Cache-Control Header中讨论的 Expires 。其它方法是:
- gzip组件
- 减少DNS查找
- 压缩JS
- 避免跳转
- 设置ETags
5. Post-load Components ——[ 延迟加载组件 ]
JavaScript是理想的(延迟)候选者,可以切分到onload
事件之前和之后。比如拖放的js库可以延迟,因为拖动必须在页面初始化之后。其它可延迟的包括隐藏的内容,折叠起来的图片等等。6. Preload Components ——[ 预加载组件 ]
预加载看起来与延迟加载相反,但它的确有个不同的目标。通过预加载你可以利用浏览器的空闲时间来请求你将来会用到的组件。这样当用户访问下一个页面时,你会有更多的组件已经在缓存中,这样会极大加快页面加载。
有几种预加载类型: - 无条件预加载:一旦onload触发,你立即获取另外的组件。比如谷歌会在主页这样加载搜索结果页面用到的雪碧图。
- 有条件预加载:基于用户动作,你推测用户下一步会去哪里并加载相应组件。
- 预期的预加载:在发布重新设计(的网站)前提前加载。在旧网页预加载新网页的部分组件,那么切换到新网页时就不会是没有任何缓存了。
7. Reduce the Number of DOM Elements ——[ 减少DOM数 ]
如果你的页面dom元素很多,那么意味着你可能需要删除无用的内容和标签来优化。8. Split Components Across Domains
把组件分配到不同的域名
把组件分散到不同的域名允许你最大化并行下载数。由于DNS查询的副作用,最佳的不同域名数是2-4。
9. Minimize the Number of iframes
最小化iframe的数量
iframe允许html文档被插入父文档。
<iframe>
优点:
- 帮助解决缓慢的第三方内容的加载,如广告和徽章
- 安全沙盒
- 并行下载脚本
<iframe>
缺点:
- 即使空的也消耗(资源和时间)
- 阻塞了页面的onload
- 非语义化(标签)
10. No 404s ——[ 不要404 ]
http请求是昂贵的,所以发出http请求但获得没用的响应(如404)是完全不必要的,并且会降低用户体验。
一些网站会有特别的404页面提高用户体验,但这仍然会浪费服务器资源。特别坏的是当链接指向外部js但却得到404结果。这样首先会降低(占用)并行下载数,其次浏览器可能会把404响应体当作js来解析,试图从里面找出可用的东西。
2. Server
1. Use a Content Delivery Network ——[ 使用CDN ]
把你的内容发布到多个,地理上分散的服务器可以让页面加载更快。
CDN是一群不同地点的服务器,可以更高效地分发内容到用户。一些大公司有自己的CDN。
2. Add an Expires or a Cache-Control Header
加Expires
或者Cache-Control
头部。
这条规则有两个方面:
- 对静态组件:通过设置
Expires
头部来实现“永不过期”策略。 - 对动态组件:用合适的
Cache-Control
头部来帮助浏览器进行有条件请求。
页面越来越丰富,意味着更多脚本,样式,图片等等。第一次访问的用户可能需要发出多个请求,但使用Expires可以让这些组件被缓存。这避免了访问子页面时没必要的http请求。Expires一般用在图片上,但应该用在所有的组件上。
浏览器(以及代理)使用缓存来减少http请求数,加快页面加载。服务器使用http响应的Expires头部来告诉客户端一个组件可以缓存多久。比如下面:
注意:如果你设置了Expires头部,当组件更新后,你必须更改文件名。
3. Gzip Components
传输时用gzip等压缩软件
http请求或响应的传输时间可以被前端工程师显著减少。终端用户的带宽,ISP,接近对等交换点等等没法被开发团队控制,但是,压缩可以通过减少http响应的大小减少响应时间。
从HTTP/1.1开始,客户端通过http请求中的Accept-Encoding头部来提示支持的压缩:
Accept-Encoding: gzip, deflate
如果服务器看到这个头部,它可能会选用列表中的某个方法压缩响应。服务器通过Content-Encoding头部提示客户端:
Content-Encoding: gzip
gzip一般可减小响应的70%。尽可能去gzip更多(文本)类型的文件。html,脚本,样式,xml和json等等都应该被gzip,而图片,pdf等等不应该被gzip,因为它们本身已被压缩过,gzip它们只是浪费cpu,甚至增加文件大小。
4. Configure ETags
实体标记(Entity tags,ETag)是服务器和浏览器之间判断浏览器缓存中某个组件是否匹配服务器端原组件的一种机制。实体就是组件:图片,脚本,样式等等。ETag被当作验证实体的比最后更改(last-modified
)日期更高效的机制。服务器这样设置组件的ETag:
之后,如果浏览器要验证组件,它用If-None-Match头部来传ETag给服务器。如果ETag匹配,服务器返回304:
ETag的问题是它们被构造来使它们对特定的运行这个网站的服务器唯一。浏览器从一个服务器获取组件,之后向另一个服务器验证,ETag将不匹配。然而服务器集群是处理请求的通用解决方案。
如果不能解决多服务器间的ETag匹配问题,那么删除ETag可能更好。
5. Flush the Buffer Early
早一点刷新buffer(尽早给浏览器数据)。
当用户请求一个页面,服务器一般要花200-500ms来拼凑整个页面。这段时间,浏览器是空闲的(等数据返回)。在php,有个方法flush()允许你传输部分准备好的html响应给浏览器。这样的话浏览器就可以开始下载组件,而同时后台可以继续生成页面剩下的部分。这种好处更多是在忙碌的后台或轻前端网站可以看到。
一个比较好的flush的位置是在head之后,因为浏览器可以加载其中的样式和脚本文件,而后台继续生成页面剩余部分。
6. Use GET for AJAX Requests
ajax请求用get。
Yahoo! Mail团队发现当使用XMLHttpRequest,POST 被浏览器实现为两步:首先发送头部,然后发送数据。所以使用GET最好,仅用一个TCP包发送(除非cookie太多)。IE的url长度限制是2K。
POST但不提交任何数据根GET行为类似,但从语义上讲,获取数据应该用GET,提交数据到服务器用POST。
7. Avoid Empty Image src
避免空src的图片。
空src属性的图片的行为可能跟你预期的不一样。它有两种形式:
- html标签:
<img src="">
- js:
var img = new Image(); img.src = ""
;
两种都会造成同一种后果:浏览器会向你的服务器发请求。
- IE,向页面所在的目录发请求。
- Safari和Chrome,请求实际的页面。
- FireFox3及之前和Safari/Chrome一样,但从3.5开始修复问题,不再发请求。
- Opera遇到空图片src不做任何事
为什么这种行为很糟糕
- 由于发送大量的意料之外的流量,会削弱服务器,尤其那些每天pv上百万的页面。
- 浪费服务器计算周期取生成不会被浏览的页面。
- 可能会破坏用户数据。如果你在跟踪请求状态,通过cookie或其它,你可能会破坏数据。即使image的请求不会返回图片,但所有的头部数据都被浏览器读取了,包括cookie。即使剩下的响应体被丢弃,破坏可能已经发生。
这种行为的根源是uri解析发生在浏览器。RFC 3986 定义了这种行为,空字符串被当作相对路径,Firefox, Safari, 和 Chrome都正确解析,而IE错误。总之,浏览器解析空字符串为相对路径的行为被认为是符合预期的。
html5在4.8.2添加了对标签src属性的描述,指导浏览器不要发出额外的请求。
The src attribute must be present, and must contain a valid URL referencing a non-interactive, optionally animated, image resource that is neither paged nor scripted. If the base URI of the element is the same as the document’s address, then the src attribute’s value must not be the empty string.
幸运的是将来浏览器不会有这个问题了(在图片上)。不幸的是,<script src="">
和<link href="">
没有这样的规范。
3. Cookie
1. Reduce Ccookie Size
http cookie的使用有多种原因,比如授权和个性化。cookie的信息通过http头部在浏览器和服务器端交换。尽可能减小cookie的大小来降低响应时间。
- 消除不必要的cookie。
- 尽可能减小cookie的大小来降低响应时间。
- 注意设置cookie到合适的域名级别,则其它子域名不会被影响。
- 正确设置Expires日期。早一点的Expires日期或者没有会尽早删除cookie,优化响应时间。
2. Use Cookie-free Domains for Components
用没有cookie的域名提供组件。
当浏览器请求静态图片并把cookie一起发送到服务器时,cookie此时对服务器没什么用处。所以这些cookie只是增加了网络流量。所以你应该保证静态组件的请求是没有cookie的。可以创建一个子域名来托管所有静态组件。
比如,你域名是www.example.org
,可以把静态组件托管在static.example.org
。不过,你如果把cookie设置在顶级域名example.org
下,这些cookie仍然会被传给static.example.org
。这种情况下,启用一个全新的域名来托管静态组件。
另外一个用没有cookie的域名提供组件的好处是,某些代理可能会阻止缓存待cookie的静态组件请求。
4. CSS
1. Put Stylesheets at the Top
研究雅虎网页性能时发现把样式表移到<head>
里会让页面更快。这是因为把样式表移到<head>
里允许页面逐步渲染。
关注性能的前端工程师希望页面被逐步渲染,这对大页面和网速慢的用户很重要
HTML规范清楚表明样式应该在<head>
里。
2. Avoid CSS Expressions
避免CSS表达式
3. Choose <link>
over @import
选择<link>
而不是@import
。点我看两者的区别
4. Avoid Filters
避免使用(IE)过滤器
IE专有的AlphaImageLoader
过滤器用于修复IE7以下版本的半透明真彩色PNG的问题。这个过滤器的问题是它阻止了渲染,并在图片下载时冻结了浏览器。另外它还引起内存消耗,并且它被应用到每个元素而不是每个图片,所以问题(的严重性)翻倍了。
最佳做法是放弃`AlphaImageLoader
,改用PNG8
来优雅降级。
5. JavaScript
#####1. Put Scripts at the Bottom
把脚本放到底部。
脚本引起的问题是它们阻塞了并行下载。HTTP1.1规范建议浏览器每个域名下不要一次下载超过2个组件。如果你的图片分散在不同服务器,那么你能并行下载多个图片。但当脚本在下载,浏览器不会再下载其它组件,即使在不同域名下。
有些情况下把脚本移动到底部并不简单。比如,脚本中用了document.write
来插入内容,它就不能被移动到底部。另外有可能有作用域问题。但大多数情况,有方法可以解决这些问题。
一个替代建议是使用异步脚本。defer
属性表明脚本不包含document.write
,是提示浏览器继续渲染的线索。不幸的是,Firefox不支持。如果脚本能异步,那么也就可以移动到底部。
2. Make JavaScript and CSS External
使用外部JS和CSS
3. Minify JavaScript and CSS
压缩JS和CSS(使用gulp或者Webpack等前端构建工具均可)
除了外部脚本和样式,內连的脚本和样式同样应该被压缩。
4. Remove Duplicate Scripts
删除重复的脚本
一种避免多次引入脚本的方法是在模板系统实现一个脚本管理模块。
5. Minimize DOM Access
最小化DOM访问。
用JS访问DOM元素是缓慢的,所以为了响应更好的页面,你应该:
有时候页面看起来不那么响应(响应速度慢),是因为绑定到不同元素的大量事件处理函数执行太多次。这是为什么使用_事件委托_是一种好方法。
另外,你不必等到onload
事件来开始处理DOM树,DOMContentLoaded
更快。大多时候你需要的只是想访问的元素已在DOM树中,所以你不必等到所有图片被下载。
6. Images
1. Optimize Images –[ 优化图片 ]
在设计师建好图片后,在上传图片到服务器前你仍可以做些事:
- 检查gif图片的调色板大小是否匹配图片颜色数。
- 可以把gif转成png看看有没有变小。除了动画,gif一般可以转成png8。
- 运行
pngcrush
或其它工具压缩png。 - 运行
jpegtran
或其它工具压缩jpeg。2. Optimize CSS Sprites –[ 优化CSS雪碧图 ]
- 把图片横向合并而不是纵向,横向更小。
- 把颜色近似的图片合并到一张雪碧图,这样可以让颜色数更少,如果低于256就可以用png8.
"Be mobile-friendly"
并且合并时图片间的间距不要太大。这对图片大小影响不是太大,但客户端解压时需要的内存更少。100×100是10000个像素,1000×1000是1000000个像素。3. Don’t Scale Images in HTML
不要再html中缩放图片
不要因为你可以设置图片的宽高就去用比你需要的大得多的图片。如果你需要:<img width="100" height="100" src="mycat.jpg" alt="My Cat" />
那么,就用100x100px的图片,而不是500x500px的.4. Make favicon.ico Small and Cacheable
favicon.ico小且缓存
favicon.ico是在你服务器根路径的图片。邪恶的是即使你不关心它,浏览器仍然会请求它。所以最好不要响应404。另外由于在同一服务器,每次请求favicon.ico时也会带上cookie。这个图片还会影响下载顺序,比如在IE,如果你在onload时下载额外的组件,fcvicon会在这些组件之前被下载。
怎么减轻favicon.ico的缺点?
#####1. Keep Components under 25K
保持组件小于25K
这个限制与iPhone不缓存大于25K的组件相关。注意,这是非压缩(uncompressed)的文件大小。在这里minification(压缩,不要与compress混淆)很重要,因为gzip无法满足(iPhone)。
2. Pack Components into a Multipart Document
打包组件到一个多部父文档
打包组件到一个多部父文档类似于带附件的邮件。它帮助你在一个http请求中获取多个组件,但注意,iPhone不支持。
看一送一
具体的关于优化的知识点或问题
1. 为什么网页设置缓存后仍然有请求(304响应)?
浏览器刷新是conditional request
,所以如果通过刷新来看缓存是否有效肯定是304。可以试试输入网址按回车或者回退键来看效果。另外由于HTML文档很少设置完全缓存(一般要和服务器验证),可以看静态组件的缓存效果(200 ok (from cache))。
2. expirationTime = responseTime + freshnessLifetime - currentAge
freshnessLifetime具体怎么算可以参考https://developer.mozilla.org/en-US/docs/Web/HTTP/Caching_FAQ
。
3. Flash of unstyled content(无样式内容闪烁)
Flash of unstyled content(FOUC)
就是在加载外部样式表之前,浏览器按默认样式显示了内容,这是因为浏览器在所有资源都下载好前就开始渲染页面了。一旦外部样式被加载,浏览器就会修正样式,但这种修正可能是可见的,也就是FOUC
。
怎么避免?在
中通过引入样式,避免使用@import。