順便提示一下:
詞法作用域:變量的作用域是在定義時決定而不是執行時決定,也就是說詞法作用域取決於源碼,通過靜態分析就能確定,因此詞法作用域也叫做靜態作用域。 with和eval除外,所以只能說JS的作用域機制非常接近詞法作用域(Lexical scope)。
下面是一個簡單的使用全局變量的閉包實例:
復制代碼 代碼如下:
var sWord="Hello,Welcome to web前端開發工程師的博客,請多多指教。"
function disWord(){
alert(sWord);
}
disWord();
解析:腳本載入到內存的時候,disWord並沒有計算sWord的值,而是函數disWord調用的時候執行了sWord的計算。
下面是函數中定義另一個函數的閉包實例:
復制代碼 代碼如下:
var iNum=10;
function add(num1,num2){
function doAdd(){return num1+num2+iNum;}
return doAdd();
}
解析:內部函數doAdd是個閉包,它將獲取傳入參數num1,num2和全局變量iNum的值,doAdd不接受參數,add最後一步調用doAdd,請兩個參數和全局變量求和返回,可以看得出doAdd使用的值是在執行環境中獲得的。
下面是在網上找的幾個例子,理解詞法作用域和閉包
復制代碼 代碼如下:
、案例一
/*全局(window)域下的一段代碼*/
function a(i) {
var i;
alert(i);
};
a(10);
疑問:上面的代碼會輸出什麼呢?
答案:10。
具體執行過程
a 函數有一個形參 i,調用 a 函數時傳入實參 10,形參 i=10
接著定義一個同名的局部變量 i,未賦值
alert 輸出 10
思考:局部變量 i 和形參 i 是同一個存儲空間嗎?
、案例二
復制代碼 代碼如下:
1 /*全局(window)域下的一段代碼*/
2 function a(i) {
3 alert(i);
4 alert(arguments[0]); //arguments[0]應該就是形參 i
5 var i = 2;
6 alert(i);
7 alert(arguments[0]);
8 };
9 a(10);
疑問:上面的代碼又會輸出什麼呢?
答案:10,10,2,2
具體執行過程
函數有一個形參i,調用 a 函數時傳入實參 10,形參 i=10
第一個 alert 把形參 i 的值 10 輸出
第二個 alert 把 arguments[0] 輸出,應該也是 i
接著定義個局部變量 i 並賦值為2,這時候局部變量 i=2
第三個 alert 就把局部變量 i 的值 2 輸出
第四個alert再次把 argumentsa[0] 輸出
思考:這裡能說明局部變量 i 和形參 i 的值相同嗎?
、案例三
復制代碼 代碼如下:
/*全局(window)域下的一段代碼*/
function a(i) {
var i = i;
alert(i);
};
a(10)
疑問:上面的代碼又又會輸出什麼呢?
答案:10
具體執行過程
第一句聲明一個與形參 i 同名的局部變量 i,根據結果我們知道,後一個 i 是指向了
形參 i,所以這裡就等於把形參 i 的值 10 賦了局部變量 i
第二個 alert 當然就輸出 10
思考:結合案列二,這裡基本能說明局部變量 i 和形參 i 指向了同一個存儲地址!
、案例四
復制代碼 代碼如下:
/*全局(window)域下的一段代碼*/
var i=10;
function a() {
alert(i);
var i = 2;
alert(i);
};
a();
疑問:上面的代碼又會輸出什麼呢?
答案:undefined, 2
具體執行過程
第一個alert輸出undefined
第二個alert輸出 2
思考:到底怎麼回事兒?
看到上面的幾個例子,你可能會想到底是怎麼執行的呢?執行的細節又是怎麼樣的呢? JS 引擎的工作方式是怎樣的呢?
解析過程
、執行順序
編譯型語言,編譯步驟分為:詞法分析、語法分析、語義檢查、代碼優化和字節生成。
解釋型語言,通過詞法分析和語法分析得到語法分析樹後,就可以開始解釋執行了。這裡是一個簡單原始的關於解析過程的原理,僅作為參考,詳細的解析過程(各種JS引擎還有不同)還需要更深一步的研究
javascript的執行過程,如果一個文檔流中包含多個script代碼段(用script標簽分隔的js代碼或引入的js文件),它們的運行順序是:
步驟1. 讀入第一個代碼段(js執行引擎並非一行一行地執行程序,而是一段一段地分析執行的)
步驟2. 做詞法分析和語法分析,有錯則報語法錯誤(比如括號不匹配等),並跳轉到步驟5
步驟3. 對【var】變量和【function】定義做“預解析“(永遠不會報錯的,因為只解析正確的聲明)
步驟4. 執行代碼段,有錯則報錯(比如變量未定義)
步驟5. 如果還有下一個代碼段,則讀入下一個代碼段,重復步驟2
步驟6. 結束
、特殊說明
全局域(window)域下所有JS代碼可以被看成是一個“匿名方法“,它會被自動執行,而此“匿名方法“內的其它方法則是在被顯示調用的時候才被執行
、關鍵步驟
上面的過程,我們主要是分成兩個階段
解析:就是通過語法分析和預解析構造合法的語法分析樹。
執行:執行具體的某個function,JS引擎在執行每個函數實例時,都會創建一個執行環境(ExecutionContext)和活動對象(activeObject)(它們屬於宿主對象,與函數實例的生命周期保持一致)
在這裡有更詳細的實例分析資料:http://www.jb51.net/article/24547.htm