亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

深入解讀JavaScript中的Hoisting機制

2019-11-20 11:50:34
字體:
來源:轉載
供稿:網友

hoisting機制

javascript的變量聲明具有hoisting機制,JavaScript引擎在執行的時候,會把所有變量的聲明都提升到當前作用域的最前面。

先看一段代碼

var v = "hello";(function(){ console.log(v); var v = "world";})();

這段代碼運行的結果是什么呢?
答案是:undefined
這段代碼說明了兩個問題,
第一,function作用域里的變量v遮蓋了上層作用域變量v。代碼做少些變動

var v = "hello";if(true){ console.log(v); var v = "world";}  

輸出結果為”hello”,說明javascript是沒有塊級作用域的。函數是JavaScript中唯一擁有自身作用域的結構。

第二,在function作用域內,變量v的聲明被提升了。所以最初的代碼相當于:

var v = "hello";(function(){ var v; //declaration hoisting console.log(v); v = "world";})();  

聲明、定義與初始化

聲明宣稱一個名字的存在,定義則為這個名字分配存儲空間,而初始化則是為名字分配的存儲空間賦初值。
用C++來表述這三個概念

extern int i;//這是聲明,表明名字i在某處已經存在了
int i;//這是聲明并定義名字i,為i分配存儲空間
i = 0;//這是初始化名字i,為其賦初值為0
javascript中則是這樣

var v;//聲明變量v
v = "hello";//(定義并)初始化變量v
因為javascript為動態語言,其變量并沒有固定的類型,其存儲空間大小會隨初始化與賦值而變化,所以其變量的“定義”就不像傳統的靜態語言一樣了,其定義顯得無關緊要。

聲明提升

當前作用域內的聲明都會提升到作用域的最前面,包括變量和函數的聲明

(function(){ var a = "1"; var f = function(){}; var b = "2"; var c = "3";})();  

 
變量a,f,b,c的聲明會被提升到函數作用域的最前面,類似如下:

(function(){ var a,f,b,c; a = "1"; f = function(){}; b = "2"; c = "3";})();

請注意函數表達式并沒有被提升,這也是函數表達式與函數聲明的區別。進一步看二者的區別:

(function(){ //var f1,function f2(){}; //hoisting,被隱式提升的聲明 f1(); //ReferenceError: f1 is not defined f2(); var f1 = function(){}; function f2(){}})();  

上面代碼中函數聲明f2被提升,所以在前面調用f2是沒問題的。雖然變量f1也被提升,但f1提升后的值為undefined,其真正的初始值是在執行到函數表達式處被賦予的。所以只有聲明是被提升的。

名字解析順序

javascript中一個名字(name)以四種方式進入作用域(scope),其優先級順序如下:
1、語言內置:所有的作用域中都有 this 和 arguments 關鍵字
2、形式參數:函數的參數在函數作用域中都是有效的
3、函數聲明:形如function foo() {}
4、變量聲明:形如var bar;

名字聲明的優先級如上所示,也就是說如果一個變量的名字與函數的名字相同,那么函數的名字會覆蓋變量的名字,無論其在代碼中的順序如何。但名字的初始化卻是按其在代碼中書寫的順序進行的,不受以上優先級的影響??创a:

(function(){  var foo;  console.log(typeof foo); //function  function foo(){}  foo = "foo";  console.log(typeof foo); //string})();  

如果形式參數中有多個同名變量,那么最后一個同名參數會覆蓋其他同名參數,即使最后一個同名參數并沒有定義。

以上的名字解析優先級存在例外,比如可以覆蓋語言內置的名字arguments。

命名函數表達式

可以像函數聲明一樣為函數表達式指定一個名字,但這并不會使函數表達式成為函數聲明。命名函數表達式的名字不會進入名字空間,也不會被提升。

f();//TypeError: f is not a function
foo();//ReferenceError: foo is not defined
var f = function foo(){console.log(typeof foo);};
f();//function
foo();//ReferenceError: foo is not defined
命名函數表達式的名字只在該函數的作用域內部有效。

再來看看下面例子:

var myval = "my global var";(function() { console.log(myval); // log "my global var"})();

以上代碼很顯然會輸出 "my global var",但是如果我們把以上代碼按如下方式稍加修改:

var myval = "my global var";(function() { console.log(myval); // log "undefined" var myval = "my local var";})();

執行結果是輸出了一個 undefined,出現這個結果的原因就是變量的聲明被提升了,以上代碼等同如下:

var myval = "my global var";(function() { var myval; console.log(myval); // log "undefined" myval = "my local var";})();

被提升的僅僅是變量的聲明部分,并沒有立即初始化,所以會輸出 undefined。

然而這種提升機制,不僅僅表現于在普通的變量,同時也表現在函數上。例如下面這段代碼并不能被正確執行:

(function() { fun(); // Uncaught TypeError: undefined is not a function var fun = function() { console.log("Hello!"); }})();

因為它等價于:

(function() { var fun; fun(); // Uncaught TypeError: undefined is not a function fun = function() { console.log("Hello!"); }})();

因為函數的聲明同樣被提升而沒有立即初始化,所以會出錯。

當然,這種定義函數的方式稱之為“函數表達式”,會有提升機制,如果是如下的這種“函數聲明”方式,則完全沒有提升機制方面的問題:

(function() { fun(); function fun() { console.log("Hello!"); // log "Hello!" }})();

這也是函數聲明與函數表達式的主要區別。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美片一区二区三区| 精品国产一区二区三区久久狼5月| 亚洲欧美在线免费| 欧美电影在线观看完整版| 国产精品久久二区| 日韩精品免费观看| 日韩成人在线网站| 97色在线播放视频| 88国产精品欧美一区二区三区| 日韩欧美成人精品| 久久久久久久久国产精品| 欧美一级电影免费在线观看| 琪琪亚洲精品午夜在线| 日韩激情视频在线| 高清欧美电影在线| 亚洲精品久久在线| 最近2019好看的中文字幕免费| 在线看日韩av| 日韩免费中文字幕| 国产精品久久不能| 1769国产精品| 成人啪啪免费看| 在线观看免费高清视频97| 亚洲精品色婷婷福利天堂| 国产精品∨欧美精品v日韩精品| 欧美性理论片在线观看片免费| 国产91精品黑色丝袜高跟鞋| 欧美巨大黑人极品精男| 亚洲精品一区av在线播放| 久久久久久久亚洲精品| 久久久久久久电影一区| 成人欧美一区二区三区黑人孕妇| 成人妇女免费播放久久久| 韩国三级电影久久久久久| 国产精品一区二区久久| 亚洲精品美女在线观看| 久久精品成人欧美大片| 亚洲天堂男人的天堂| 在线观看日韩专区| 超碰97人人做人人爱少妇| 国产精品v片在线观看不卡| 久久精品亚洲一区| 亚洲久久久久久久久久久| 日韩高清免费在线| 欧美激情视频在线| 日韩在线视频中文字幕| 色播久久人人爽人人爽人人片视av| 26uuu另类亚洲欧美日本老年| 国产一区二区av| 国产精品流白浆视频| 97国产精品人人爽人人做| 欧美电影第一页| 国产精品免费久久久久久| 欧美日韩国产精品专区| 久久免费少妇高潮久久精品99| 欧美另类交人妖| 深夜福利国产精品| 日韩福利视频在线观看| 777777777亚洲妇女| 亚洲xxxx视频| 亚洲视频在线免费看| 欧美一区二区三区免费观看| 久久久久久久国产精品视频| 2019中文字幕免费视频| 日本亚洲欧美三级| 午夜精品美女自拍福到在线| 国产精品欧美日韩| 在线观看精品自拍私拍| 久久久亚洲网站| 国产视频久久久久| 欧美日韩免费一区| 欧美日韩国产一区二区三区| 狠狠躁18三区二区一区| 国产成人亚洲综合| 亚洲精品一区在线观看香蕉| 日本韩国欧美精品大片卡二| 国产精品久久久av| 欧美性猛交xxxx偷拍洗澡| 97欧美精品一区二区三区| 亚洲va欧美va在线观看| 国产精品视频男人的天堂| 久久久久999| 久久中文字幕国产| 日韩成人在线视频网站| 黄色一区二区三区| 日本最新高清不卡中文字幕| 欧美成人第一页| 97在线免费观看视频| 主播福利视频一区| 亚洲国产精品高清久久久| 日韩在线观看高清| 不卡av电影在线观看| 最近2019中文字幕在线高清| 国产欧美日韩中文字幕在线| 欧美日韩国产丝袜另类| 操91在线视频| 精品国产老师黑色丝袜高跟鞋| 国产欧美精品日韩| 欧美性猛交xxxx黑人| 日韩精品视频在线播放| 日韩欧美高清在线视频| 992tv成人免费视频| 亚洲成av人片在线观看香蕉| 一区国产精品视频| 91在线精品播放| 欧美高清视频免费观看| 2018日韩中文字幕| 亚洲精品一区二区久| 国产午夜精品视频免费不卡69堂| 亚洲人成绝费网站色www| 91精品国产综合久久香蕉最新版| 2019中文字幕在线免费观看| 日韩精品中文字幕在线观看| 91精品在线播放| 国产在线精品成人一区二区三区| xxxx欧美18另类的高清| 久久久av亚洲男天堂| 亚洲国产精品成人一区二区| 欧美性生活大片免费观看网址| www.亚洲男人天堂| 国产精品视频不卡| 久久精品成人欧美大片古装| 国产性猛交xxxx免费看久久| 色伦专区97中文字幕| 日韩欧美精品在线观看| 久久中文精品视频| 久久久久久久久久久久久久久久久久av| 国产一区二区三区免费视频| 精品视频—区二区三区免费| 中文字幕欧美专区| 国产精品一二三在线| 国产精品久久国产精品99gif| 亚洲精选一区二区| 91国内精品久久| 欧美高跟鞋交xxxxxhd| 欧美性少妇18aaaa视频| 亚洲一区二区三区乱码aⅴ| 欧美日韩成人在线播放| 国产亚洲一级高清| 日韩美女在线观看一区| 国产午夜精品美女视频明星a级| 久久免费视频在线观看| 欧美性xxxx在线播放| 国产亚洲精品久久久久久777| 欧美黑人国产人伦爽爽爽| 久久久亚洲国产| 亚洲男人天堂久| 亚洲人成电影在线| 国产成一区二区| 最近中文字幕2019免费| 欧美日韩亚洲国产一区| 亚洲精品suv精品一区二区| 欧美猛少妇色xxxxx| 久久精品小视频| 国产成人精品一区| 亲爱的老师9免费观看全集电视剧| 亚洲成年网站在线观看| 91免费视频网站| 国产偷国产偷亚洲清高网站| 国模吧一区二区三区| 午夜精品久久久久久99热| 两个人的视频www国产精品| 黑人欧美xxxx| 国产视频久久久|