标签:
本来这是个老生常谈的问题,上周自成又分享了一些性能优化的建议,我这里再做一个全面的Tips整理,谨作为查阅型的文档,不妥之处,还请指正;
一、 Yahoo的规则条例:
谨记:80%-90%的终端响应时间是花费在下载页面中的图片,样式表,脚本,flash等;
详细的解释来这里查:http://developer.yahoo.com/performance/rules.html
也可以直接firebug上一项项比对,如下图:
简单翻译解释下:
1、尽量减少HTTP请求个数——须权衡
合并图片(如css sprites,内置图片使用数据)、合并CSS、JS,这一点很重要,但是要考虑合并后的文件体积。
2、使用CDN(内容分发网络)
这里可以关注CDN的三类实现:镜像、高速缓存、专线,以及智能路由器和负载均衡;
3、为文件头指定Expires或Cache-Control,使内容具有缓存性。
区分静态内容和动态内容,避免以后页面访问中不必要的HTTP请求。
4、避免空的src和href
留意具有这两个属性的标签如link,script,img,iframe等;
5、使用gzip压缩内容
Gzip压缩所有可能的文件类型以来减少文件体积
6、把CSS放到顶部
实现页面有秩序地加载,这对于拥有较多内容的页面和网速较慢的用户来说更为重要,同时,HTML规范清楚指出样式表要放包含在页面的区域内;
7、把JS放到底部
HTTP/1.1 规范建议,浏览器每个主机名的并行下载内容不超过两个,而问题在于脚本阻止了页面的平行下载,即便是主机名不相同
8、避免使用CSS表达式
页面显示和缩放,滚动、乃至移动鼠标时,CSS表达式的计算频率是我们要关注的。可以考虑一次性的表达式或者使用事件句柄来代替CSS表达式。
9、将CSS和JS放到外部文件中
我们需要权衡内置代码带来的HTTP请求减少与通过使用外部文件进行缓存带来的好处的折中点。
10、减少DNS查找次数
我们需要权衡减少 DNS查找次数和保持较高程度并行下载两者之间的关系。
11、精简CSS和JS
目的就是减少下载的文件体积,可考虑压缩工具JSMin和YUI Compressor。
12、避免跳转
为了确保“后退”按钮可以正确地使用,使用标准的 3XXHTTP状态代码;同域中注意避免反斜杠 “/” 的跳转;
跨域使用 Alias或者 mod_rewirte建立 CNAME(保存一个域名和另外一个域名之间关系的DNS记录)
13、剔除重复的JS和CSS
重复调用脚本,除了增加额外的HTTP请求外,多次运算也会浪费时间。在IE和Firefox中不管脚本是否可缓存,它们都存在重复运算JavaScript的问题。
14、配置ETags
Entity
tags(ETags)(实体标签)是web服务器和浏览器用于判断浏览器缓存中的内容和服务器中的原始内容是否匹配的一种机制(“实体”就是所说的“内
容”,包括图片、脚本、样式表等),是比last-modified
date更更加灵活的机制,单位时间内文件被修过多次,Etag可以综合Inode(文件的索引节点(inode)数),MTime(修改时间)和
Size来精准的进行判断,避开UNIX记录MTime只能精确到秒的问题。
服务器集群使用,可取后两个参数。使用ETags减少Web应用带宽和负载。
15、使AJAX可缓存
利用时间戳,更精巧的实现响应可缓存与服务器数据同步更新。
16、尽早刷新输出缓冲
尤其对于css,js文件的并行下载更有意义
17、使用GET来完成AJAX请求
当使用XMLHttpRequest时,浏览器中的POST方法是一个“两步走”的过程:首先发送文件头,然后才发送数据。在url小于2K时使用GET获取数据时更加有意义。
18、延迟加载
确定页面运行正常后,再加载脚本来实现如拖放和动画,或者是隐藏部分的内容以及折叠内容等。
19、预加载
关注下无条件加载,有条件加载和有预期的加载。
20、减少DOM元素个数
使用更适合或者在语意是更贴切的标签,要考虑大量DOM元素中循环的性能开销。
21、根据域名划分页面内容
很显然, 是最大限度地实现平行下载
22、尽量减少iframe的个数
考虑即使内容为空,加载也需要时间,会阻止页面加载,没有语意,注意iframe相对于其他DOM元素高出1-2个数量级的开销,它会在典型方式下阻塞onload事件,IE和Firefox中主页面样式表会阻塞它的下载。
23、避免404
HTTP请求时间消耗是很大的,有些站点把404错误响应页面改为“你是不是要找***”,这虽然改进了用户体验但是同样也会浪费服务器资源(如数
据库等)。最糟糕的情况是指向外部
JavaScript的链接出现问题并返回404代码。首先,这种加载会破坏并行加载;其次浏览器会把试图在返回的404响应内容中找到可能有用的部分当
作JavaScript代码来执行。
24、减少Cookie的大小
25、使用无cookie的域
确定对于静态内容的请求是无coockie的请求。创建一个子域名并用他来存放所有静态内容。
26、减少DOM访问
27、开发智能事件处理程序
有时候我们会感觉到页面反应迟钝,这是因为DOM树元素中附加了过多的事件句柄并且些事件句病被频繁地触发。这就是为什么说使用event
delegation(事件代理)是一种好方法了。如果你在一个div中有10个按钮,你只需要在div上附加一次事件句柄就可以了,而不用去为每一个按
钮增加一个句柄。事件冒泡时你可以捕捉到事件并判断出是哪个事件发出的。
你同样也不用为了操作DOM树而等待onload事件的发生。你需要做的就是等待树结构中你要访问的元素出现。你也不用等待所有图像都加载完毕。
你可能会希望用DOMContentLoaded事件来代替 事件应用程序中的onAvailable方法。
28、用 代替@import
在IE中,页面底部@import和使用作用是一样的,因此最好不要使用它。
29、避免使用滤镜
完全避免使用AlphaImageLoader的最好方法就是使用PNG8格式来代替,这种格式能在IE中很好地工作。如果你确实需要使用 AlphaImageLoader,请使用下划线_filter又使之对IE7以上版本的用户无效。
30、优化图像
尝试把GIF格式转换成PNG格式,看看是否节省空间。在所有的PNG图片上运行pngcrush(或者其它PNG优化工具)
31、优化CSS Spirite
在Spirite中水平排列你的图片,垂直排列会稍稍增加文件大小;
Spirite中把颜色较近的组合在一起可以降低颜色数,理想状况是低于256色以便适用PNG8格式;
便于移动,不要在Spirite的图像中间留有较大空隙。这虽然不大会增加文件大小但对于用户代理来说它需要更少的内存来把图片解压为像素地图。 100×100的图片为1万像素,而1000×1000就是100万像素。
32、不要在HTML中缩放图像——须权衡
不要为了在HTML中设置长宽而使用比实际需要大的图片。如果你需要:
1
|
< img width=”100″ height=”100″ src=”mycat.jpg” alt=”My Cat” /> |
那么你的图片(mycat.jpg)就应该是100×100像素而不是把一个500×500像素的图片缩小使用。这里在下文有更有趣的分析。
33、favicon.ico要小而且可缓存
favicon.ico是位于服务器根目录下的一个图片文件。它是必定存在的,因为即使你不关心它是否有用,浏览器也会对它发出请求,因此最好不要
返回一 个404 Not
Found的响应。由于是在同一台服务器上,它每被请求一次coockie就会被发送一次。这个图片文件还会影响下载顺序,例如在IE中当你在
onload中请求额外的文件时,favicon会在这些额外内容被加载前下载。
因此,为了减少favicon.ico带来的弊端,要做到:
文件尽量地小,最好小于1K
在适当的时候(也就是你不要打算再换favicon.ico的时候,因为更换新文件时不能对它进行重命名)为它设置Expires文件头。你可以很安全地 把Expires文件头设置为未来的几个月。你可以通过核对当前favicon.ico的上次编辑时间来作出判断。
Imagemagick可以帮你创建小巧的favicon。
34、保持单个内容小于25K
因为iPhone不能缓存大于25K的文件。注意这里指的是解压缩后的大小。由于单纯gizp压缩可能达不要求,因此精简文件就显得十分重要。
35、打包组件成复合文本
页面内容打包成复合文本就如同带有多附件的Email,它能够使你在一个HTTP请求中取得多个组件(切记:HTTP请求是很奢侈的)。当你使用这条规 则时,首先要确定用户代理是否支持(iPhone就不支持)。
二、Yahoo军规之外的场景?
1、 使用json作为数据的交换格式
Json在浏览器解析的效率至少高于XML一个数量级,高级浏览器中内置的有生成和解析json的方法,IE6中要用额外的方法(http://json.org ),不要用eval,容易引发性能和安全问题。
2、 尽可能对images和table设定宽高值
针对Yslow的不要在HTML中缩放图像——第33条,有人会误解为不要对图片加宽高值,其实这条建议本身的意思是不要为了获取一个特定大小的图片,而去强行通过设置宽高值拉伸或者压缩一个既有的图片。建议是另存一张符合尺寸的图片替代。
对图片和table是设定宽高,是考虑到如果浏览器能立刻知道图片或者tables的宽高,它就能够直接呈现页面而不需要通过计算元素大小后重绘,而且即便是图片损毁而没有展现,也不会进而破坏了页面本来的布局。
有一些应用场景需要注意:
1 2 <img width=”100″ height=”120″ src=”" alt=”" />
c、批量图片,图片源不可控,页面图片宽高值不定,比如数据库有100张各种尺寸偏差较大的,此时可不对图片设置宽高;
其他情况不一一罗列,原则是在最大程度保证图片不变形与图片最大面积展现的前提下,尽可能为图片设置宽高值,总之就是权衡。
Tables的宽高值同图片,尽可能设置。
3、 拆离内容块
尽量用div取代tables,或者将tables打破成嵌套层次深的结构;
避免用这样的嵌套:
1 <table> 2 <table> 3 <table> 4 ... 5 </table> 6 </table> 7 </table>
采用下面的或者div重构:
1 <table></table> 2 <table></table> 3 <table></table>
4、 高效的CSS书写规则
众所周知,CSS选择符是从右向左进行匹配的。
通常一个图片列表的的小模块:
1 <div id="box"> 2 <div class="hd"> 3 <h3>我的旅途</h3> 4 </div> 5 <div class="bd"> 6 <h4>旅途1</h4> 7 <ul id="pics"> 8 <li> 9 <a href="#pic" title=""><img src="" alt="" /> </a> 10 <p>这是在<strong>图片1</strong></p> 11 </li> 12 </ul> 13 </div> 14 </div>
<span class="Apple-style-span" style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;">为了代码上缩进后内层的整洁性,我们html有可能这样写之外,更喜欢看这样的css写法:</span>
1 .box{border:1px solid #ccc } 2 .box .hd{border-bottom:1px solid #ccc } 3 .box .hd h3{color:#515151} 4 .box .bd{color:#404040 } 5 .box .bd ul{margin-left:10px} 6 .box .bd ul li{border-bottom:1px dashed #f1f1f1} 7 .box .bd ul li a{text-decoration:none} 8 .box .bd ul li a:hover{text-decoration:underline} 9 .box .bd ul li a img{border:1px solid #ccc} 10 .box .bd ul li p{text-align:left;} 11 .box .bd ul li p strong{color:#ff6600}
<span class="Apple-style-span" style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;">其实写到这里,问题已经显而易见了。深达五层抑或六层的嵌套,同时右边的选择符都是采用标签,在满足我们视觉平整与代码结构系统化的时候,
付出的是性能的代价。</span>
不做进一步的代码书写方式的探讨,受个人习惯与应用场景影响。这里对css选择符按照开销从小到大的顺序梳理一下:
伪类和伪元素 a:hover
参考《高性能网站建设-进阶指南》,有如下建议:
还要注意到,即便是页面加载后,当页面被触发引起回流(reflow)的时候,低效的选择符依然会引发更高的开销,显然这对于用户是不佳的体验。
4、Javascript 的性能优化点
1 var tips = ‘tip1‘+‘tip2‘;
这是我们拼接字符串常用的方式,但是这种方式会有一些临时变量的创建和销毁,影响性能,尤其是在IE6下,所以推荐使用如下方式拼接:
1 var tip_array = [],tips; 2 tip_array.push(‘tip1‘); 3 tip_array.push(‘tip2‘); 4 tips = tip_array.join(‘‘);
当然,最新的浏览器(如火狐 Firefox3+,IE8+ 等等)对字符串的拼接做了优化,性能略快于数组的“join”方法。
5、DOM 操作优化
首先澄清两个概念——Repaint 和 Reflow:Repaint 也叫 Redraw,它指的是一种不会影响当前 DOM 的结构和布局的一种重绘动作。如下动作会产生 Repaint 动作:
Reflow 比起 Repaint 来讲就是一种更加显著的变化了。它主要发生在 DOM 树被操作的时候,任何改变 DOM 的结构和布局都会产生 Reflow。但一个元素的 Reflow 操作发生时,它的所有父元素和子元素都会放生 Reflow,最后 Reflow 必然会导致 Repaint 的产生。举例说明,如下动作会产生 Reflow 动作:
浏览器窗口的变化;
DOM 节点的添加删除操作
一些改变页面元素大小,形状和位置的操作的触发通过 Reflow 和 Repaint 的介绍可知,每次 Reflow 比其 Repaint 会带来更多的资源消耗,因此,我们应该尽量减少 Reflow 的发生,或者将其转化为只会触发 Repaint 操作的代码。
1 var tipBox = document.createElement(‘div‘); 2 document.body.appendChild(‘tipBox‘);//reflow 3 var tip1 = document.createElement(‘div‘); 4 var tip2 = document.createElement(‘div‘); 5 tipBox.appendChild(tip1);//reflow 6 tipBox.appendChild(tip2);//reflow
<span class="Apple-style-span" style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;">如上的代码,会产生三次reflow,优化后的代码如下:</span>
1 var tipBox = document.createElement(‘div‘); 2 tip1 = document.createElement(‘div‘); 3 tip2 = document.createElement(‘div‘); 4 tipBox.appendChild(tip1); 5 tipBox.appendChild(tip2); 6 document.body.appendChild(‘tipBox‘);//reflow
<span class="Apple-style-span" style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;">当然还可以利用 display 来减少reflow次数</span>
1 var tipBox = document.getElementById(‘tipBox‘); 2 tipBox.style.display = ‘none‘;//reflow 3 tipBox.appendChild(tip1); 4 tipBox.appendChild(tip2); 5 tipBox.appendChild(tip3); 6 tipBox.appendChild(tip4); 7 tipBox.appendChild(tip5); 8 tipBox.style.width = 120; 9 tipBox.style.height = 60; 10 tipBox.style.display = ‘block‘;//reflow
<span class="Apple-style-span" style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;">DOM元素测量属性和方法也会触发reflow,如下:</span>
1 2 var tipWidth = tipBox.offsetWidth;//reflow 3 tipScrollLeft = tipBox.scrollLeft;//reflow 4 display = window.getComputedStyle(div,‘‘).getPropertyValue(‘display‘);//reflow
<span class="Apple-style-span" style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;">触发reflow的属性和方法大概有这些:</span>
clientTop/Left/Width/Height
我们可以用临时变量将“offsetWidth”的值缓存起来,这样就不用每次访问“offsetWidth”属性。这种方式在循环里面非常适用,可以极大地提高性能。
如果有批量的样式属性需要修改,建议通过替换className的方式来降低reflow的次数,曾经有这样一个场景:有三个intput,分别对
应下面三个图片和三个内容区域,第二input选中的时候,第二图片显示,其他图片隐藏,第二块内容显示,其他内容隐藏,直接操作DOM节点的代码如下:
1 var input = []; 2 pics = []; 3 contents = []; 4 ...... 5 inputFrame.onclick =function(e){ 6 var _e,_target; 7 _e = e ? window.event : null; 8 if(!_e){ 9 return; 10 }else{ 11 _target = _e.srcElement || _e.target ; 12 _index = getIndex(_target);//reflow两次 13 show(_target,_index);//reflow两次 14 } 15 16 } 17 function show(target,j){ 18 for(var i = 0,i<3;i++){ 19 target[i].style.display = ‘none‘;//reflow 20 } 21 target[j].style.display = ‘block‘;//reflow 22 } 23 function getIndex(targer){ 24 if(target){ 25 .....//获取当前的元素索引 26 return index; 27 } 28 }
<span class="Apple-style-span" style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;">如果是通过css预先定义元素的隐藏和显示,通过对父级的className进行操纵,将会把reflow的次数减少到1次</span>
1 .pbox .pic,.pbox content{display:none} 2 .J_pbox_0 .pic0,.J_pbox_0 .content0{diplay:block} 3 .J_pbox_1 .pic1,.J_pbox_1 .content1{diplay:block} 4 .J_pbox_2 .pic2,.J_pbox_2 .content2{diplay:block}
<strong style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px; line-height: 19px; white-space: normal;"></strong>
1 var input = [], 2 parentBox = document.getELementById(‘J_Pbox‘); 3 ...... 4 inputFrame.onclick =function(e){ 5 var _e,_target; 6 if(){ 7 ... 8 }else{ 9 ... 10 parentBox.className = ‘pbox J_pbox_‘+_infex;//reflow一次 11 } 12 }
<strong style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;"> </strong>
<strong style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px;
line-height: 19px; white-space: normal;"> </strong>
<strong style="font-family: Georgia, ‘Times New Roman‘, ‘Bitstream Charter‘, Times, serif; font-size: 13px; line-height: 19px;
white-space: normal;">三、Yahoo军规再度挖掘会怎样?</strong>
在网站性能优化的路上,是不会有终点的,这也是前端工程师永不会妥协的地方。
想看到更牛P的优化建议么,请移步这里来关注李牧童鞋的分享:
原文:alimama ued
标签:
原文地址:http://www.cnblogs.com/jsStudyjj/p/4220404.html