JavaScript 性能優(yōu)化的小知識(shí)總結(jié)

前言

一直在學(xué)習(xí)javascript,也有看過(guò)《犀利開(kāi)發(fā)Jquery內(nèi)核詳解與實(shí)踐》,對(duì)這本書(shū)的評(píng)價(jià)只有兩個(gè)字犀利,可能是對(duì)javascript理解的還不夠透徹異或是自己太笨,更多的是自己不擅于思考懶得思考以至于里面說(shuō)的一些精髓都沒(méi)有太深入的理解。

鑒于想讓自己有一個(gè)提升,進(jìn)不了一個(gè)更加廣闊的天地,總得找一個(gè)屬于自己的居所好好生存,所以平時(shí)會(huì)有意無(wú)意的去積累一些使用jQuerry的常用知識(shí),特別是對(duì)于性能要求這一塊,總是會(huì)想是不是有更好的方式來(lái)實(shí)現(xiàn)。

下面是我總結(jié)的一些小技巧,僅供參考。(我先會(huì)說(shuō)一個(gè)總標(biāo)題,然后用一小段話來(lái)說(shuō)明這個(gè)意思 再最后用一個(gè)demo來(lái)簡(jiǎn)單言明)

避免全局查找

在一個(gè)函數(shù)中會(huì)用到全局對(duì)象存儲(chǔ)為局部變量來(lái)減少全局查找,因?yàn)樵L問(wèn)局部變量的速度要比訪問(wèn)全局變量的速度更快些

functionsearch(){

//當(dāng)我要使用當(dāng)前頁(yè)面地址和主機(jī)域名

alert(window.location.href+window.location.host);

}

//最好的方式是如下這樣??先用一個(gè)簡(jiǎn)單變量保存起來(lái)

functionsearch(){

varlocation=window.location;

alert(location.href+location.host);

}

定時(shí)器

如果針對(duì)的是不斷運(yùn)行的代碼,不應(yīng)該使用setTimeout,而應(yīng)該是用setInterval,因?yàn)閟etTimeout每一次都會(huì)初始化一個(gè)定時(shí)器,而setInterval只會(huì)在開(kāi)始的時(shí)候初始化一個(gè)定時(shí)器

vartimeoutTimes=0;

functiontimeout(){

timeoutTimes++;

if(timeoutTimes<10){

setTimeout(timeout,10);

}

}

timeout();

//可以替換為:

varintervalTimes=0;

functioninterval(){

intervalTimes++;

if(intervalTimes>=10){

clearInterval(interv);

}

}

varinterv=setInterval(interval,10);

字符串連接

如果要連接多個(gè)字符串,應(yīng)該少使用+=,如

s+=a;

s+=b;

s+=c;

應(yīng)該寫(xiě)成s+=a + b + c;

而如果是收集字符串,比如多次對(duì)同一個(gè)字符串進(jìn)行+=操作的話,最好使用一個(gè)緩存,使用JavaScript數(shù)組來(lái)收集,最后使用join方法連接起來(lái)

varbuf=[];

for(vari=0;i<100;i++){

buf.push(i.toString());

}

varall=buf.join("");

避免with語(yǔ)句

和函數(shù)類(lèi)似 ,with語(yǔ)句會(huì)創(chuàng)建自己的作用域,因此會(huì)增加其中執(zhí)行的代碼的作用域鏈的長(zhǎng)度,由于額外的作用域鏈的查找,在with語(yǔ)句中執(zhí)行的代碼肯定會(huì)比外面執(zhí)行的代碼要慢,在能不使用with語(yǔ)句的時(shí)候盡量不要使用with語(yǔ)句

with(a.b.c.d){

property1=1;

property2=2;

}

//可以替換為:

varobj=a.b.c.d;

obj.property1=1;

obj.property2=2;

數(shù)字轉(zhuǎn)換成字符串

般最好用”” + 1來(lái)將數(shù)字轉(zhuǎn)換成字符串,雖然看起來(lái)比較丑一點(diǎn),但事實(shí)上這個(gè)效率是最高的,性能上來(lái)說(shuō):

(“” +) > String() > .toString() > new String()

浮點(diǎn)數(shù)轉(zhuǎn)換成整型

很多人喜歡使用parseInt(),其實(shí)parseInt()是用于將字符串轉(zhuǎn)換成數(shù)字,而不是浮點(diǎn)數(shù)和整型之間的轉(zhuǎn)換,我們應(yīng)該使用Math.floor()或者M(jìn)ath.round()

各種類(lèi)型轉(zhuǎn)換

varmyVar="3.14159",

str=""+myVar,//??to string

i_int=~~myVar,//??to integer

f_float=1*myVar,//??to float

b_bool=!!myVar,/*??to boolean - any string with length

and any number except 0 are true */

array=[myVar];//??to array

如果定義了toString()方法來(lái)進(jìn)行類(lèi)型轉(zhuǎn)換的話,推薦顯式調(diào)用toString(),因?yàn)閮?nèi)部的操作在嘗試所有可能性之后,會(huì)嘗試對(duì)象的toString()方法嘗試能否轉(zhuǎn)化為String,所以直接調(diào)用這個(gè)方法效率會(huì)更高

多個(gè)類(lèi)型聲明

在JavaScript中所有變量都可以使用單個(gè)var語(yǔ)句來(lái)聲明,這樣就是組合在一起的語(yǔ)句,以減少整個(gè)腳本的執(zhí)行時(shí)間,就如上面代碼一樣,上面代碼格式也挺規(guī)范,讓人一看就明了。

插入迭代器

如var name=values[i]; i++;前面兩條語(yǔ)句可以寫(xiě)成var name=values[i++]

使用直接量

varaTest=newArray();//替換為

varaTest=[];

varaTest=newObject;//替換為

varaTest={};

varreg=newRegExp();//替換為

varreg=/../;

//如果要?jiǎng)?chuàng)建具有一些特性的一般對(duì)象,也可以使用字面量,如下:

varoFruit=newO;

oFruit.color="red";

oFruit.name="apple";

//前面的代碼可用對(duì)象字面量來(lái)改寫(xiě)成這樣:

varoFruit={color:"red",name:"apple"};

使用DocumentFragment優(yōu)化多次append

一旦需要更新DOM,請(qǐng)考慮使用文檔碎片來(lái)構(gòu)建DOM結(jié)構(gòu),然后再將其添加到現(xiàn)存的文檔中。

for(vari=0;i<1000;i++){

varel=document.createElement('p');

el.innerHTML=i;

document.body.appendChild(el);

}

//可以替換為:

varfrag=document.createDocumentFragment();

for(vari=0;i<1000;i++){

varel=document.createElement('p');

el.innerHTML=i;

frag.appendChild(el);

}

document.body.appendChild(frag);

使用一次innerHTML賦值代替構(gòu)建dom元素

對(duì)于大的DOM更改,使用innerHTML要比使用標(biāo)準(zhǔn)的DOM方法創(chuàng)建同樣的DOM結(jié)構(gòu)快得多。

JavaScript

varfrag=document.createDocumentFragment();

for(vari=0;i<1000;i++){

varel=document.createElement('p');

el.innerHTML=i;

frag.appendChild(el);

}

document.body.appendChild(frag);

//可以替換為:

varhtml=[];

for(vari=0;i<1000;i++){

html.push('

'+i+'

');

}

document.body.innerHTML=html.join('');

通過(guò)模板元素clone,替代createElement

很多人喜歡在JavaScript中使用document.write來(lái)給頁(yè)面生成內(nèi)容。事實(shí)上這樣的效率較低,如果需要直接插入HTML,可以找一個(gè)容器元素,比如指定一個(gè)div或者span,并設(shè)置他們的innerHTML來(lái)將自己的HTML代碼插入到頁(yè)面中。通常我們可能會(huì)使用字符串直接寫(xiě)HTML來(lái)創(chuàng)建節(jié)點(diǎn),其實(shí)這樣做,1無(wú)法保證代碼的有效性2字符串操作效率低,所以應(yīng)該是用document.createElement()方法,而如果文檔中存在現(xiàn)成的樣板節(jié)點(diǎn),應(yīng)該是用cloneNode()方法,因?yàn)槭褂胏reateElement()方法之后,你需要設(shè)置多次元素的屬性,使用cloneNode()則可以減少屬性的設(shè)置次數(shù)——同樣如果需要?jiǎng)?chuàng)建很多元素,應(yīng)該先準(zhǔn)備一個(gè)樣板節(jié)點(diǎn)

ag=document.createDocumentFragment();

for(vari=0;i<1000;i++){

varel=document.createElement('p');

el.innerHTML=i;

frag.appendChild(el);

}

document.body.appendChild(frag);

//替換為:

varfrag=document.createDocumentFragment();

varpEl=document.getElementsByTagName('p')[0];

for(vari=0;i<1000;i++){

varel=pEl.cloneNode(false);

el.innerHTML=i;

frag.appendChild(el);

}

document.body.appendChild(frag);

使用firstChild和nextSibling代替childNodes遍歷dom元素


varnodes=element.childNodes;

for(vari=0,l=nodes.length;i<1;i++){

varnode=nodes[i];

//……

}

//可以替換為:

varnode=element.firstChild;

while(node){

//……

node=node.nextSibling;

刪除DOM節(jié)點(diǎn)

刪除dom節(jié)點(diǎn)之前,一定要?jiǎng)h除注冊(cè)在該節(jié)點(diǎn)上的事件,不管是用observe方式還是用attachEvent方式注冊(cè)的事件,否則將會(huì)產(chǎn)生無(wú)法回收的內(nèi)存。另外,在removeChild和innerHTML=’’二者之間,盡量選擇后者. 因?yàn)樵趕IEve(內(nèi)存泄露監(jiān)測(cè)工具)中監(jiān)測(cè)的結(jié)果是用removeChild無(wú)法有效地釋放dom節(jié)點(diǎn)

使用事件代理

任何可以冒泡的事件都不僅僅可以在事件目標(biāo)上進(jìn)行處理,目標(biāo)的任何祖先節(jié)點(diǎn)上也能處理,使用這個(gè)知識(shí)就可以將事件處理程序附加到更高的地方負(fù)責(zé)多個(gè)目標(biāo)的事件處理,同樣,對(duì)于內(nèi)容動(dòng)態(tài)增加并且子節(jié)點(diǎn)都需要相同的事件處理函數(shù)的情況,可以把事件注冊(cè)提到父節(jié)點(diǎn)上,這樣就不需要為每個(gè)子節(jié)點(diǎn)注冊(cè)事件監(jiān)聽(tīng)了。另外,現(xiàn)有的js庫(kù)都采用observe方式來(lái)創(chuàng)建事件監(jiān)聽(tīng),其實(shí)現(xiàn)上隔離了dom對(duì)象和事件處理函數(shù)之間的循環(huán)引用,所以應(yīng)該盡量采用這種方式來(lái)創(chuàng)建事件監(jiān)聽(tīng)

重復(fù)使用的調(diào)用結(jié)果,事先保存到局部變量

//避免多次取值的調(diào)用開(kāi)銷(xiāo)

varh1=element1.clientHeight+num1;

varh4=element1.clientHeight+num2;

//可以替換為:

vareleHeight=element1.clientHeight;

varh1=eleHeight+num1;

varh4=eleHeight+num2;

注意NodeList

最小化訪問(wèn)NodeList的次數(shù)可以極大的改進(jìn)腳本的性能

varimages=document.getElementsByTagName('img');

for(vari=0,len=images.length;i

}

編寫(xiě)JavaScript的時(shí)候一定要知道何時(shí)返回NodeList對(duì)象,這樣可以最小化對(duì)它們的訪問(wèn)

進(jìn)行了對(duì)getElementsByTagName()的調(diào)用

獲取了元素的childNodes屬性

獲取了元素的attributes屬性

訪問(wèn)了特殊的集合,如document.forms、document.images等等

要了解了當(dāng)使用NodeList對(duì)象時(shí),合理使用會(huì)極大的提升代碼執(zhí)行速度

優(yōu)化循環(huán)

可以使用下面幾種方式來(lái)優(yōu)化循環(huán)

減值迭代

大多數(shù)循環(huán)使用一個(gè)從0開(kāi)始、增加到某個(gè)特定值的迭代器,在很多情況下,從最大值開(kāi)始,在循環(huán)中不斷減值的迭代器更加高效

簡(jiǎn)化終止條件

由于每次循環(huán)過(guò)程都會(huì)計(jì)算終止條件,所以必須保證它盡可能快,也就是說(shuō)避免屬性查找或者其它的操作,最好是將循環(huán)控制量保存到局部變量中,也就是說(shuō)對(duì)數(shù)組或列表對(duì)象的遍歷時(shí),提前將length保存到局部變量中,避免在循環(huán)的每一步重復(fù)取值。

varlist=document.getElementsByTagName('p');

for(vari=0;i

//……

}

//替換為:

varlist=document.getElementsByTagName('p');

for(vari=0,l=list.length;i

//……

}

簡(jiǎn)化循環(huán)體

循環(huán)體是執(zhí)行最多的,所以要確保其被最大限度的優(yōu)化

使用后測(cè)試循環(huán)

在JavaScript中,我們可以使用for(;;),while(),for(in)三種循環(huán),事實(shí)上,這三種循環(huán)中for(in)的效率極差,因?yàn)樗枰樵?xún)散列鍵,只要可以,就應(yīng)該盡量少用。for(;;)和while循環(huán),while循環(huán)的效率要優(yōu)于for(;;),可能是因?yàn)閒or(;;)結(jié)構(gòu)的問(wèn)題,需要經(jīng)常跳轉(zhuǎn)回去。


vararr=[1,2,3,4,5,6,7];

varsum=0;

for(vari=0,l=arr.length;i

sum+=arr[i];

}

//可以考慮替換為:

vararr=[1,2,3,4,5,6,7];

varsum=0,l=arr.length;

while(l--){

sum+=arr[l];

}

最常用的for循環(huán)和while循環(huán)都是前測(cè)試循環(huán),而如do-while這種后測(cè)試循環(huán),可以避免最初終止條件的計(jì)算,因此運(yùn)行更快。

展開(kāi)循環(huán)

當(dāng)循環(huán)次數(shù)是確定的,消除循環(huán)并使用多次函數(shù)調(diào)用往往會(huì)更快。

避免雙重解釋

如果要提高代碼性能,盡可能避免出現(xiàn)需要按照J(rèn)avaScript解釋的字符串,也就是

盡量少使用eval函數(shù)

使用eval相當(dāng)于在運(yùn)行時(shí)再次調(diào)用解釋引擎對(duì)內(nèi)容進(jìn)行運(yùn)行,需要消耗大量時(shí)間,而且使用Eval帶來(lái)的安全性問(wèn)題也是不容忽視的。

不要使用Function構(gòu)造器

不要給setTimeout或者setInterval傳遞字符串參數(shù)

varnum=0;

setTimeout('num++',10);

//可以替換為:

varnum=0;

functionaddNum(){

num++;

}

setTimeout(addNum,10);

縮短否定檢測(cè)

if(oTest!='#ff0000'){

//do something

}

if(oTest!=null){

//do something

}

if(oTest!=false){

//do something

}

//雖然這些都正確,但用邏輯非操作符來(lái)操作也有同樣的效果:

if(!oTest){

//do something

}

條件分支

將條件分支,按可能性順序從高到低排列:可以減少解釋器對(duì)條件的探測(cè)次數(shù)

在同一條件子的多(>2)條件分支時(shí),使用switch優(yōu)于if:switch分支選擇的效率高于if,在IE下尤為明顯。4分支的測(cè)試,IE下switch的執(zhí)行時(shí)間約為if的一半。

使用三目運(yùn)算符替代條件分支

if(a>b){

num=a;

}else{

num=b;

}

//可以替換為:

num=a>b?a:b;

使用常量

重復(fù)值:任何在多處用到的值都應(yīng)該抽取為一個(gè)常量

用戶(hù)界面字符串:任何用于顯示給用戶(hù)的字符串,都應(yīng)該抽取出來(lái)以方便國(guó)際化

URLs:在Web應(yīng)用中,資源位置很容易變更,所以推薦用一個(gè)公共地方存放所有的URL

任意可能會(huì)更改的值:每當(dāng)你用到字面量值的時(shí)候,你都要問(wèn)一下自己這個(gè)值在未來(lái)是不是會(huì)變化,如果答案是“是”,那么這個(gè)值就應(yīng)該被提取出來(lái)作為一個(gè)常量。

避免與null進(jìn)行比較

由于JavaScript是弱類(lèi)型的,所以它不會(huì)做任何的自動(dòng)類(lèi)型檢查,所以如果看到與null進(jìn)行比較的代碼,嘗試使用以下技術(shù)替換

如果值應(yīng)為一個(gè)引用類(lèi)型,使用instanceof操作符檢查其構(gòu)造函數(shù)

如果值應(yīng)為一個(gè)基本類(lèi)型,作用typeof檢查其類(lèi)型

如果是希望對(duì)象包含某個(gè)特定的方法名,則使用typeof操作符確保指定名字的方法存在于對(duì)象上

避免全局量

全局變量應(yīng)該全部字母大寫(xiě),各單詞之間用_下劃線來(lái)連接。盡可能避免全局變量和函數(shù), 盡量減少全局變量的使用,因?yàn)樵谝粋€(gè)頁(yè)面中包含的所有JavaScript都在同一個(gè)域中運(yùn)行。所以如果你的代碼中聲明了全局變量或者全局函數(shù)的話,后面的代碼中載入的腳本文件中的同名變量和函數(shù)會(huì)覆蓋掉(overwrite)你的。

//糟糕的全局變量和全局函數(shù)

varcurrent=null;

functioninit(){

//...

}

functionchange(){

//...

}

functionverify(){

//...

}

//解決辦法有很多,Christian Heilmann建議的方法是:

//如果變量和函數(shù)不需要在“外面”引用,那么就可以使用一個(gè)沒(méi)有名字的方法將他們?nèi)及饋?lái)。

(function(){

varcurrent=null;

functioninit(){

//...

}

functionchange(){

//...

}

functionverify(){

//...

}

})();

//如果變量和函數(shù)需要在“外面”引用,需要把你的變量和函數(shù)放在一個(gè)“命名空間”中

//我們這里用一個(gè)function做命名空間而不是一個(gè)var,因?yàn)樵谇罢咧新暶鱢unction更簡(jiǎn)單,而且能保護(hù)隱私數(shù)據(jù)

myNameSpace=function(){

varcurrent=null;

functioninit(){

//...

}

functionchange(){

//...

}

functionverify(){

//...

}

//所有需要在命名空間外調(diào)用的函數(shù)和屬性都要寫(xiě)在return里面

return{

init:init,

//甚至你可以為函數(shù)和屬性命名一個(gè)別名

set:change

};

};

尊重對(duì)象的所有權(quán)

因?yàn)镴avaScript可以在任何時(shí)候修改任意對(duì)象,這樣就可以以不可預(yù)計(jì)的方式覆寫(xiě)默認(rèn)的行為,所以如果你不負(fù)責(zé)維護(hù)某個(gè)對(duì)象,它的對(duì)象或者它的方法,那么你就不要對(duì)它進(jìn)行修改,具體一點(diǎn)就是說(shuō):

不要為實(shí)例或原型添加屬性

不要為實(shí)例或者原型添加方法

不要重定義已經(jīng)存在的方法

不要重復(fù)定義其它團(tuán)隊(duì)成員已經(jīng)實(shí)現(xiàn)的方法,永遠(yuǎn)不要修改不是由你所有的對(duì)象,你可以通過(guò)以下方式為對(duì)象創(chuàng)建新的功能:

創(chuàng)建包含所需功能的新對(duì)象,并用它與相關(guān)對(duì)象進(jìn)行交互

創(chuàng)建自定義類(lèi)型,繼承需要進(jìn)行修改的類(lèi)型,然后可以為自定義類(lèi)型添加額外功能

循環(huán)引用

如果循環(huán)引用中包含DOM對(duì)象或者ActiveX對(duì)象,那么就會(huì)發(fā)生內(nèi)存泄露。內(nèi)存泄露的后果是在瀏覽器關(guān)閉前,即使是刷新頁(yè)面,這部分內(nèi)存不會(huì)被瀏覽器釋放。

簡(jiǎn)單的循環(huán)引用:

varel=document.getElementById('MyElement');

varfunc=function(){

//…

}

el.func=func;

func.element=el;

但是通常不會(huì)出現(xiàn)這種情況。通常循環(huán)引用發(fā)生在為dom元素添加閉包作為expendo的時(shí)候。


functioninit(){

varel=document.getElementById('MyElement');

el.onclick=function(){

//……

}

}

init();

init在執(zhí)行的時(shí)候,當(dāng)前上下文我們叫做context。這個(gè)時(shí)候,context引用了el,el引用了function,function引用了context。這時(shí)候形成了一個(gè)循環(huán)引用。

下面2種方法可以解決循環(huán)引用:

1)置空dom對(duì)象

functioninit(){

varel=document.getElementById('MyElement');

el.onclick=function(){

//……

}

}

init();

//可以替換為:

functioninit(){

varel=document.getElementById('MyElement');

el.onclick=function(){

//……

}

el=null;

}

init();

將el置空,context中不包含對(duì)dom對(duì)象的引用,從而打斷循環(huán)應(yīng)用。

如果我們需要將dom對(duì)象返回,可以用如下方法:

functioninit(){

varel=document.getElementById('MyElement');

el.onclick=function(){

//……

}

returnel;

}

init();

//可以替換為:

functioninit(){

varel=document.getElementById('MyElement');

el.onclick=function(){

//……

}

try{

returnel;

}finally{

el=null;

}

}

init();

2)構(gòu)造新的context

functioninit(){

varel=document.getElementById('MyElement');

el.onclick=function(){

//……

}

}

init();

//可以替換為:

functionelClickHandler(){

//……

}

functioninit(){

varel=document.getElementById('MyElement');

el.onclick=elClickHandler;

}

init();

把function抽到新的context中,這樣,function的context就不包含對(duì)el的引用,從而打斷循環(huán)引用。

通過(guò)javascript創(chuàng)建的dom對(duì)象,必須append到頁(yè)面中

IE下,腳本創(chuàng)建的dom對(duì)象,如果沒(méi)有append到頁(yè)面中,刷新頁(yè)面,這部分內(nèi)存是不會(huì)回收的!

functioncreate(){

vargc=document.getElementById('GC');

for(vari=0;i<5000;i++){

varel=document.createElement('div');

el.innerHTML="test";

//下面這句可以注釋掉,看看瀏覽器在任務(wù)管理器中,點(diǎn)擊按鈕然后刷新后的內(nèi)存變化

gc.appendChild(el);

}

}

釋放dom元素占用的內(nèi)存

將dom元素的innerHTML設(shè)置為空字符串,可以釋放其子元素占用的內(nèi)存。

在rich應(yīng)用中,用戶(hù)也許會(huì)在一個(gè)頁(yè)面上停留很長(zhǎng)時(shí)間,可以使用該方法釋放積累得越來(lái)越多的dom元素使用的內(nèi)存。

釋放javascript對(duì)象

在rich應(yīng)用中,隨著實(shí)例化對(duì)象數(shù)量的增加,內(nèi)存消耗會(huì)越來(lái)越大。所以應(yīng)當(dāng)及時(shí)釋放對(duì)對(duì)象的引用,讓GC能夠回收這些內(nèi)存控件。

對(duì)象:obj = null

對(duì)象屬性:delete obj.myproperty

數(shù)組item:使用數(shù)組的splice方法釋放數(shù)組中不用的item

避免string的隱式裝箱

對(duì)string的方法調(diào)用,比如’xxx’.length,瀏覽器會(huì)進(jìn)行一個(gè)隱式的裝箱操作,將字符串先轉(zhuǎn)換成一個(gè)String對(duì)象。推薦對(duì)聲明有可能使用String實(shí)例方法的字符串時(shí),采用如下寫(xiě)法:

var myString = new String(‘Hello World’);

松散耦合

1、解耦HTML/JavaScript

JavaScript和HTML的緊密耦合:直接寫(xiě)在HTML中的JavaScript、使用包含內(nèi)聯(lián)代碼的

HTML和JavaScript的緊密耦合:JavaScript中包含HTML,然后使用innerHTML來(lái)插入一段html文本到頁(yè)面

其實(shí)應(yīng)該是保持層次的分離,這樣可以很容易的確定錯(cuò)誤的來(lái)源,所以我們應(yīng)確保HTML呈現(xiàn)應(yīng)該盡可能與JavaScript保持分離

2、解耦CSS/JavaScript

顯示問(wèn)題的唯一來(lái)源應(yīng)該是CSS,行為問(wèn)題的唯一來(lái)源應(yīng)該是JavaScript,層次之間保持松散耦合才可以讓你的應(yīng)用程序更加易于維護(hù),所以像以下的代碼element.style.color=”red”盡量改為element.className=”edit”,而且不要在css中通過(guò)表達(dá)式嵌入JavaScript

3、解耦應(yīng)用程序/事件處理程序

將應(yīng)用邏輯和事件處理程序相分離:一個(gè)事件處理程序應(yīng)該從事件對(duì)象中提取,并將這些信息傳送給處理應(yīng)用邏輯的某個(gè)方法中。這樣做的好處首先可以讓你更容易更改觸發(fā)特定過(guò)程的事件,其次可以在不附加事件的情況下測(cè)試代碼,使其更易創(chuàng)建單元測(cè)試

性能方面的注意事項(xiàng)

1、盡量使用原生方法

2、switch語(yǔ)句相對(duì)if較快

通過(guò)將case語(yǔ)句按照最可能到最不可能的順序進(jìn)行組織

3、位運(yùn)算較快

當(dāng)進(jìn)行數(shù)字運(yùn)算時(shí),位運(yùn)算操作要比任何布爾運(yùn)算或者算數(shù)運(yùn)算快

4、巧用||和&&布爾運(yùn)算符

functioneventHandler(e){

if(!e)e=window.event;

}

//可以替換為:

functioneventHandler(e){

e=e||window.event;

}

if(myobj){

doSomething(myobj);

}

//可以替換為:

myobj&&doSomething(myobj);

避免錯(cuò)誤應(yīng)注意的地方

1、每條語(yǔ)句末尾須加分號(hào)

在if語(yǔ)句中,即使條件表達(dá)式只有一條語(yǔ)句也要用{}把它括起來(lái),以免后續(xù)如果添加了語(yǔ)句之后造成邏輯錯(cuò)誤

2、使用+號(hào)時(shí)需謹(jǐn)慎

JavaScript 和其他編程語(yǔ)言不同的是,在 JavaScript 中,’+’除了表示數(shù)字值相加,字符串相連接以外,還可以作一元運(yùn)算符用,把字符串轉(zhuǎn)換為數(shù)字。因而如果使用不當(dāng),則可能與自增符’++’混淆而引起計(jì)算錯(cuò)誤

varvalueA=20;

varvalueB="10";

alert(valueA+valueB);//ouput: 2010

alert(valueA+(+valueB));//output: 30

alert(valueA++valueB);//output:30

alert(valueA++valueB);//Compile error

3、使用return語(yǔ)句需要注意

一條有返回值的return語(yǔ)句不要用()括號(hào)來(lái)括住返回值,如果返回表達(dá)式,則表達(dá)式應(yīng)與return關(guān)鍵字在同一行,以避免壓縮時(shí),壓縮工具自動(dòng)加分號(hào)而造成返回與開(kāi)發(fā)人員不一致的結(jié)果

functionF1(){

varvalueA=1;

varvalueB=2;

returnvalueA+valueB;

}

functionF2(){

varvalueA=1;

varvalueB=2;

return

valueA+valueB;

}

alert(F1());//output: 3

alert(F2());//ouput: undefined

==和===的區(qū)別

避免在if和while語(yǔ)句的條件部分進(jìn)行賦值,如if (a = b),應(yīng)該寫(xiě)成if (a == b),但是在比較是否相等的情況下,最好使用全等運(yùn)行符,也就是使用===和!==操作符會(huì)相對(duì)于==和!=會(huì)好點(diǎn)。==和!=操作符會(huì)進(jìn)行類(lèi)型強(qiáng)制轉(zhuǎn)換

varvalueA="1";

varvalueB=1;

if(valueA==valueB){

alert("Equal");

}

else{

alert("Not equal");

}

//output: "Equal"

if(valueA===valueB){

alert("Equal");

}

else{

alert("Not equal");

}

//output: "Not equal"

不要使用生偏語(yǔ)法

不要使用生偏語(yǔ)法,寫(xiě)讓人迷惑的代碼,雖然計(jì)算機(jī)能夠正確識(shí)別并運(yùn)行,但是晦澀難懂的代碼不方便以后維護(hù)

函數(shù)返回統(tǒng)一類(lèi)型

雖然JavaScript是弱類(lèi)型的,對(duì)于函數(shù)來(lái)說(shuō),前面返回整數(shù)型數(shù)據(jù),后面返回布爾值在編譯和運(yùn)行都可以正常通過(guò),但為了規(guī)范和以后維護(hù)時(shí)容易理解,應(yīng)保證函數(shù)應(yīng)返回統(tǒng)一的數(shù)據(jù)類(lèi)型

總是檢查數(shù)據(jù)類(lèi)型

要檢查你的方法輸入的所有數(shù)據(jù),一方面是為了安全性,另一方面也是為了可用性。用戶(hù)隨時(shí)隨地都會(huì)輸入錯(cuò)誤的數(shù)據(jù)。這不是因?yàn)樗麄兇?,而是因?yàn)樗麄兒苊?,并且思考的方式跟你不同。用typeof方法來(lái)檢測(cè)你的function接受的輸入是否合法

何時(shí)用單引號(hào),何時(shí)用雙引號(hào)

雖然在JavaScript當(dāng)中,雙引號(hào)和單引號(hào)都可以表示字符串, 為了避免混亂,我們建議在HTML中使用雙引號(hào),在JavaScript中使用單引號(hào),但為了兼容各個(gè)瀏覽器,也為了解析時(shí)不會(huì)出錯(cuò),定義JSON對(duì)象時(shí),最好使用雙引號(hào)

部署

用JSLint運(yùn)行JavaScript驗(yàn)證器來(lái)確保沒(méi)有語(yǔ)法錯(cuò)誤或者是代碼沒(méi)有潛在的問(wèn)

部署之前推薦使用壓縮工具將JS文件壓縮

文件編碼統(tǒng)一用UTF-8

JavaScript 程序應(yīng)該盡量放在 .js 的文件中,需要調(diào)用的時(shí)候在 HTML 中以

標(biāo)簽前。這樣會(huì)降低因加載 JavaScript 代碼而影響頁(yè)面中其它組件的加載時(shí)間。

永遠(yuǎn)不要忽略代碼優(yōu)化工作,重構(gòu)是一項(xiàng)從項(xiàng)目開(kāi)始到結(jié)束需要持續(xù)的工作,只有不斷的優(yōu)化代碼才能讓代碼的執(zhí)行效率越來(lái)越好


感興趣的同學(xué),加裙289683894 交流學(xué)習(xí),領(lǐng)取學(xué)習(xí)資料

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容