国产高清在线免费观看-国产高清在线免费无码-国产高清在线男人的天堂-国产高清在线视频-国产高清在线视频精品视频-国产高清在线视频伊甸园

LOGO OA教程 ERP教程 模切知識交流 PMS教程 CRM教程 開發(fā)文檔 其他文檔  
 
網(wǎng)站管理員

JavaScript開發(fā)進階:理解JavaScript作用域和作用域鏈

admin
2011年9月7日 14:51 本文熱度 3637

  作用域是JavaScript最重要的概念之一,想要學好JavaScript就需要理解JavaScript作用域和作用域鏈的工作原理。今天這篇文章對JavaScript作用域和作用域鏈作簡單的介紹,希望能幫助大家更好的學習JavaScript。


JavaScript作用域


  任何程序設計語言都有作用域的概念,簡單的說,作用域就是變量與函數(shù)的可訪問范圍,即作用域控制著變量與函數(shù)的可見性和生命周期。在JavaScript中,變量的作用域有全局作用域和局部作用域兩種。


  1.  全局作用域(Global Scope)


  在代碼中任何地方都能訪問到的對象擁有全局作用域,一般來說以下幾種情形擁有全局作用域:


  (1)最外層函數(shù)和在最外層函數(shù)外面定義的變量擁有全局作用域,例如:


var authorName="山邊小溪";
function doSomething(){
var blogName="夢想天空";
function innerSay(){
alert(blogName);
}
innerSay();
}
alert(authorName); //山邊小溪
alert(blogName); //腳本錯誤
doSomething(); //夢想天空
innerSay() //腳本錯誤

  (2)所有末定義直接賦值的變量自動聲明為擁有全局作用域,例如:


function doSomething(){
var authorName="山邊小溪";
blogName="夢想天空";
alert(authorName);
}
doSomething(); //山邊小溪
alert(blogName); //夢想天空
alert(authorName); //腳本錯誤

  變量blogName擁有全局作用域,而authorName在函數(shù)外部無法訪問到。


  (3)所有window對象的屬性擁有全局作用域


  一般情況下,window對象的內(nèi)置屬性都擁有全局作用域,例如window.name、window.location、window.top等等。


  1.  局部作用域(Local Scope)  


  和全局作用域相反,局部作用域一般只在固定的代碼片段內(nèi)可訪問到,最常見的例如函數(shù)內(nèi)部,所有在一些地方也會看到有人把這種作用域稱為函數(shù)作用域,例如下列代碼中的blogName和函數(shù)innerSay都只擁有局部作用域。


function doSomething(){
var blogName="夢想天空";
function innerSay(){
alert(blogName);
}
innerSay();
}
alert(blogName); //腳本錯誤
innerSay(); //腳本錯誤

作用域鏈(Scope Chain)


  在JavaScript中,函數(shù)也是對象,實際上,JavaScript里一切都是對象。函數(shù)對象和其它對象一樣,擁有可以通過代碼訪問的屬性和一系列僅供JavaScript引擎訪問的內(nèi)部屬性。其中一個內(nèi)部屬性是[[Scope]],由ECMA-262標準第三版定義,該內(nèi)部屬性包含了函數(shù)被創(chuàng)建的作用域中對象的集合,這個集合被稱為函數(shù)的作用域鏈,它決定了哪些數(shù)據(jù)能被函數(shù)訪問。


  當一個函數(shù)創(chuàng)建后,它的作用域鏈會被創(chuàng)建此函數(shù)的作用域中可訪問的數(shù)據(jù)對象填充。例如定義下面這樣一個函數(shù):


function add(num1,num2) {
var sum = num1 + num2;
return sum;
}

  在函數(shù)add創(chuàng)建時,它的作用域鏈中會填入一個全局對象,該全局對象包含了所有全局變量,如下圖所示(注意:圖片只例舉了全部變量中的一部分):



JavaScript作用域鏈



  函數(shù)add的作用域?qū)趫?zhí)行時用到。例如執(zhí)行如下代碼:


var total = add(5,10);

  執(zhí)行此函數(shù)時會創(chuàng)建一個稱為“運行期上下文(execution context)”的內(nèi)部對象,運行期上下文定義了函數(shù)執(zhí)行時的環(huán)境。每個運行期上下文都有自己的作用域鏈,用于標識符解析,當運行期上下文被創(chuàng)建時,而它的作用域鏈初始化為當前運行函數(shù)的[[Scope]]所包含的對象。


  這些值按照它們出現(xiàn)在函數(shù)中的順序被復制到運行期上下文的作用域鏈中。它們共同組成了一個新的對象,叫“活動對象(activation object)”,該對象包含了函數(shù)的所有局部變量、命名參數(shù)、參數(shù)集合以及this,然后此對象會被推入作用域鏈的前端,當運行期上下文被銷毀,活動對象也隨之銷毀。新的作用域鏈如下圖所示:



JavaScript作用域鏈



  在函數(shù)執(zhí)行過程中,沒遇到一個變量,都會經(jīng)歷一次標識符解析過程以決定從哪里獲取和存儲數(shù)據(jù)。該過程從作用域鏈頭部,也就是從活動對象開始搜索,查找同名的標識符,如果找到了就使用這個標識符對應的變量,如果沒找到繼續(xù)搜索作用域鏈中的下一個對象,如果搜索完所有對象都未找到,則認為該標識符未定義。函數(shù)執(zhí)行過程中,每個標識符都要經(jīng)歷這樣的搜索過程。


作用域鏈和代碼優(yōu)化


  從作用域鏈的結構可以看出,在運行期上下文的作用域鏈中,標識符所在的位置越深,讀寫速度就會越慢。如上圖所示,因為全局變量總是存在于運行期上下文作用域鏈的最末端,因此在標識符解析的時候,查找全局變量是最慢的。所以,在編寫代碼的時候應盡量少使用全局變量,盡可能使用局部變量。一個好的經(jīng)驗法則是:如果一個跨作用域的對象被引用了一次以上,則先把它存儲到局部變量里再使用。例如下面的代碼:


function changeColor(){
document.getElementById("btnChange").onclick=function(){
document.getElementById("targetCanvas").style.backgroundColor="red";
};
}

  這個函數(shù)引用了兩次全局變量document,查找該變量必須遍歷整個作用域鏈,直到最后在全局對象中才能找到。這段代碼可以重寫如下:


function changeColor(){
var doc=document;
doc.getElementById("btnChange").onclick=function(){
doc.getElementById("targetCanvas").style.backgroundColor="red";
};
}

  這段代碼比較簡單,重寫后不會顯示出巨大的性能提升,但是如果程序中有大量的全局變量被從反復訪問,那么重寫后的代碼性能會有顯著改善。


改變作用域鏈


  函數(shù)每次執(zhí)行時對應的運行期上下文都是獨一無二的,所以多次調(diào)用同一個函數(shù)就會導致創(chuàng)建多個運行期上下文,當函數(shù)執(zhí)行完畢,執(zhí)行上下文會被銷毀。每一個運行期上下文都和一個作用域鏈關聯(lián)。一般情況下,在運行期上下文運行的過程中,其作用域鏈只會被 with 語句和 catch 語句影響。


  with語句是對象的快捷應用方式,用來避免書寫重復代碼。例如:


function initUI(){
with(document){
var bd=body,
links=getElementsByTagName("a"),
i=0,
len=links.length;
while(i < len){
update(links[i++]);
}
getElementById("btnInit").onclick=function(){
doSomething();
};
}
}

  這里使用width語句來避免多次書寫document,看上去更高效,實際上產(chǎn)生了性能問題。


  當代碼運行到with語句時,運行期上下文的作用域鏈臨時被改變了。一個新的可變對象被創(chuàng)建,它包含了參數(shù)指定的對象的所有屬性。這個對象將被推入作用域鏈的頭部,這意味著函數(shù)的所有局部變量現(xiàn)在處于第二個作用域鏈對象中,因此訪問代價更高了。如下圖所示:



JavaScript作用域鏈



  因此在程序中應避免使用with語句,在這個例子中,只要簡單的把document存儲在一個局部變量中就可以提升性能。


  另外一個會改變作用域鏈的是try-catch語句中的catch語句。當try代碼塊中發(fā)生錯誤時,執(zhí)行過程會跳轉(zhuǎn)到catch語句,然后把異常對象推入一個可變對象并置于作用域的頭部。在catch代碼塊內(nèi)部,函數(shù)的所有局部變量將會被放在第二個作用域鏈對象中。示例代碼:


try{
doSomething();
}catch(ex){
alert(ex.message); //作用域鏈在此處改變
}

  請注意,一旦catch語句執(zhí)行完畢,作用域鏈機會返回到之前的狀態(tài)。try-catch語句在代碼調(diào)試和異常處理中非常有用,因此不建議完全避免。你可以通過優(yōu)化代碼來減少catch語句對性能的影響。一個很好的模式是將錯誤委托給一個函數(shù)處理,例如:


try{
doSomething();
}catch(ex){
handleError(ex); //委托給處理器方法
}

  優(yōu)化后的代碼,handleError方法是catch子句中唯一執(zhí)行的代碼。該函數(shù)接收異常對象作為參數(shù),這樣你可以更加靈活和統(tǒng)一的處理錯誤。由于只執(zhí)行一條語句,且沒有局部變量的訪問,作用域鏈的臨時改變就不會影響代碼性能了。


參考資料


1. High.Performance.JavaScript, Nicholas.C.Zakas


2. Explaining JavaScript Scope And Closures, Robert Nyman


3. ECMAScript Language Specification, bclary.com


該文章在 2011/9/7 14:51:41 編輯過
關鍵字查詢
相關文章
正在查詢...
點晴ERP是一款針對中小制造業(yè)的專業(yè)生產(chǎn)管理軟件系統(tǒng),系統(tǒng)成熟度和易用性得到了國內(nèi)大量中小企業(yè)的青睞。
點晴PMS碼頭管理系統(tǒng)主要針對港口碼頭集裝箱與散貨日常運作、調(diào)度、堆場、車隊、財務費用、相關報表等業(yè)務管理,結合碼頭的業(yè)務特點,圍繞調(diào)度、堆場作業(yè)而開發(fā)的。集技術的先進性、管理的有效性于一體,是物流碼頭及其他港口類企業(yè)的高效ERP管理信息系統(tǒng)。
點晴WMS倉儲管理系統(tǒng)提供了貨物產(chǎn)品管理,銷售管理,采購管理,倉儲管理,倉庫管理,保質(zhì)期管理,貨位管理,庫位管理,生產(chǎn)管理,WMS管理系統(tǒng),標簽打印,條形碼,二維碼管理,批號管理軟件。
點晴免費OA是一款軟件和通用服務都免費,不限功能、不限時間、不限用戶的免費OA協(xié)同辦公管理系統(tǒng)。
Copyright 2010-2025 ClickSun All Rights Reserved

主站蜘蛛池模板: 在线视频欧美日韩不卡一区 | 91精品国产色综合久久 | 亚洲av无码精品网站 | 国产精品猎奇系列在线观看 | 人妻无码久久久久久久久久久 | 亚洲国产精品无码久久电影 | 18岁禁看网站| 四虎影视最新的2025版地址 | 人澡人爽精品A片一区 | 国产日本精品 | 亚洲国产日韩欧美在线a乱码 | 国产成人av在线免播放观看 | 亚洲精品久久一区二区三区777 | 国产丝袜美女 | 欧美日韩亚洲综合一区二区三区 | 日本三区高清不卡 | 亚洲国产福利一区在线观看 | 人妻 校园 激情 另类 | 精品久久久久久无码人妻热 | 日韩免费在线观看国产一区二区三区 | 亚洲天堂在线免费观看视频 | 香蕉人人超人人超碰超国产 | 综合色天天鬼久久鬼色 | 欧美日韩人妻精品系列一区二区三区 | 国产成人精品一区二区三区免费 | 最新免费视频高清老师中文无码一区二区三区在线观看 | 另类在线视频 | 亚洲欧美韩国日产综合在线 | 99在线精品免费视频 | 人妻日本无中文字幕无码 | 亚洲高清毛片一区 | 一区二区三区欧美在线 | 亚洲av无码成h人动漫无遮挡 | 国产精品免费一区二区区 | 亚洲精品天堂成人片av在线播放 | 日本无码特黄午夜视频在线观看 | 少妇spa推油被扣高潮 | 无码观看欧美夜夜夜夜爽 | 国产成人久久精品二三区麻豆 | 一本一道久久a久久精品综合 | 人妻丰满熟妇av无码区不卡 |