成熟丰满熟妇高潮XXXXX,人妻无码AV中文系列久久兔费 ,国产精品一国产精品,国精品午夜福利视频不卡麻豆

您好,歡迎來到九壹網(wǎng)。
搜索
您的當(dāng)前位置:首頁JS中作用域和變量提升(hoisting)的深入理解

JS中作用域和變量提升(hoisting)的深入理解

來源:九壹網(wǎng)
作用域(Scoping)

對于Javascript初學(xué)者來說,一個(gè)最迷惑的地方就是作用域;事實(shí)上,不光是初學(xué)者。我就見過一些有經(jīng)驗(yàn)的javascript程序員,但他們對scope理解不深。javascript作用域之所以迷惑,是因?yàn)樗绦蛘Z法本身長的像C家族的語言。我對作用域的理解是只會對某個(gè)范圍產(chǎn)生作用,而不會對外產(chǎn)生影響的封閉空間。在這樣的一些空間里,外部不能訪問內(nèi)部變量,但內(nèi)部可以訪問外部變量。

c語言的變量分為全局變量和局部變量,全局變量的作用范圍是任何文件和函數(shù)訪問(當(dāng)然,對于非變量定義的其他c文件,需要使用extern關(guān)鍵字進(jìn)行申明,使用static關(guān)鍵字也可以將作用范圍限定在當(dāng)前文件中),局部變量的作用范圍就是從申明到最近的大括號涵蓋的塊級范圍。java則無全局變量,有類變量,成員變量和局部變量,作用范圍根據(jù)public,protected,private等訪問權(quán)限有不同的作用范圍,這里就不多述。

JS作用域有哪些?

在ES5中,js只有兩種形式的作用域:全局作用域和函數(shù)作用域。

全局作用域其實(shí)是全局對象的作用域,任意地方都可以訪問到(如果沒有被函數(shù)作用域覆蓋)。

函數(shù)對象作用域跟c的局部變量作用域是不同的,它的作用域是整個(gè)函數(shù)范圍,不論他是在函數(shù)的任意位置申明的!這就是所謂的hoisting,也就是變量提升的概念。不過不著急,下面會專門針對hoisting來進(jìn)行解釋。

不過,在ES6中,新增了一個(gè)塊級作用域(最近的大括號涵蓋的范圍),但是僅限于let方式申明的變量。
作用域演示:

201611011003251.png

定義變量時(shí),如果不寫var,比如 i=0,則會被定義為全局變量,作用域?yàn)槿肿饔糜?,否則為局部變量,作用域?yàn)楹瘮?shù)作用域。上面第一行的var i=0,之所以說它是全局變量,是因?yàn)樗呀?jīng)是在全局區(qū)申明的了,并不在函數(shù)范圍內(nèi),因此跟 i=0 是一樣的。

至于,為什么結(jié)果會是這樣,繼續(xù)往下看就知道了。

申明形式

變量聲明:

201611011003251.png

函數(shù)申明:

201611011003251.png

變量提升(Hoisting)

引出一個(gè)問題

下面這段代碼會輸出什么內(nèi)容?

201611011003251.png

這道題我面試過很多人,大多數(shù)人都說輸出的是日期。但真實(shí)的結(jié)果是undefined。為什么是這樣呢?這里就引出了一個(gè)概念--hoisting,中文的意思就是變量提升。MDN中對變量hoisting的解釋是這樣的:

var hoisting

Because variable declarations (and declarations in general) are processed before any code is executed, declaring a variable anywhere in the code is equivalent to declaring it at the top. This also means that a variable can appear to be used before it's declared. This behavior is called "hoisting", as it appears that the variable declaration is moved to the top of the function or global code.

這段話翻譯下來就是

因?yàn)樽兞可昝魇窃谌我獯a執(zhí)行前處理的,在代碼區(qū)中任意地方申明變量和在最開始(最上面)的地方申明是一樣的。也就是說,看起來一個(gè)變量可以在申明之前被使用!這種行為就是所謂的“hoisting”,也就是變量提升,看起來就像變量的申明被自動移動到了函數(shù)或全局代碼的最頂上。

注意:僅僅是申明提升了,定義并不會被提升。

如此,上面這段代碼其實(shí)就是下面的形式:

201611011003251.png

所以,這樣就應(yīng)該理解了,console輸出的時(shí)候,tmp變量僅僅是申明了但未定義,所以輸出應(yīng)該是undefined。

這里需要說明的是,雖然所有的申明(包括ES5的var、function,和ES6的function *、let、const、class)都會被提升,但是var、function、function *和let、const、class的的提升卻并不相同!具體原因可以看這里的說明(大體的意思是雖然let,const,class也被提升了,但是卻并不會被初始化,這時(shí)候去訪問他們則會報(bào)ReferenceError異常,他們需要到語句執(zhí)行的時(shí)候才會被初始化,而在被初始化之前的狀態(tài)叫做temporal dead zone)。我們來看一段代碼就知道了:

201611011003251.png

這里a被提升,但因?yàn)槎x在后,所以輸出undefined

這里a雖然被提升,但卻報(bào)了引用錯(cuò)誤!

之所以或這樣

因?yàn)檫@樣的原因,推薦的做法是在申明變量的時(shí)候,將所用的變量都寫在作用域(全局作用域或函數(shù)作用域)的最頂上,這樣代碼看起來就會更清晰,更容易看出來那個(gè)變量是來自函數(shù)作用域的,哪個(gè)又是來自作用域鏈(本文不對此多做解釋,請讀者自行百度,有機(jī)會再補(bǔ)充說明)。

重復(fù)聲明

201611011003251.png

上面的輸出其實(shí)是:1 2 2。雖然看起來里面x申明了兩次,但上面說了,js的var變量只有全局作用域和函數(shù)作用域兩種,且申明會被提升,因此實(shí)際上x只會在最頂上開始的地方申明一次,var x=2的申明會被忽略,僅用于賦值。也就是說上面的代碼實(shí)際上跟下面是一致的。

201611011003251.png

函數(shù)和變量同時(shí)提升的問題

如果是函數(shù)和變量類型同時(shí)申明定義了,會發(fā)生什么事情呢?看下面的代碼

201611011003251.png

上面的輸出結(jié)果其實(shí)是: function foo(){} ,也就是函數(shù)內(nèi)容。

而如果是這樣的形式呢

201611011003251.png

它的輸出卻變成:undefined

為什么會這樣呢?

原來函數(shù)提升分為兩種情況:

一種:函數(shù)申明。就是上面A,function foo(){}這種形式

另一種:函數(shù)表達(dá)式。就是上面B,var foo=function(){}這種形式

第二種形式其實(shí)就是var變量的聲明定義,因此上面的B輸出結(jié)果為undefined應(yīng)該就能理解了。

而第一種函數(shù)申明的形式,在提升的時(shí)候,會被整個(gè)提升上去,包括函數(shù)定義的部分!因此A跟下面的這種方式是等價(jià)的!

201611011003251.png

原因是因?yàn)椋?、函數(shù)聲明被提升到最頂上;2、申明只進(jìn)行一次,因此后面var foo='i am text'的申明會被忽略。

并且函數(shù)申明的優(yōu)先級優(yōu)于變量申明,所以以下形式的輸出,同樣是函數(shù)內(nèi)容:

201611011003251.png

總結(jié)

要徹底理解JS的作用域和Hoisting,只要記住以下三點(diǎn)即可:

1、所有申明都會被提升到作用域的最頂上

2、同一個(gè)變量申明只進(jìn)行一次,并且因此其他申明都會被忽略

3、函數(shù)聲明的優(yōu)先級優(yōu)于變量申明,且函數(shù)聲明會連帶定義一起被提升

注意:

通過with語句,可以臨時(shí)改變運(yùn)行期上下文的作用域鏈,此時(shí)的對非var定義的變量進(jìn)行訪問,會首先訪問with中對象的屬性,然后才會向上順著作用域鏈向上檢查該屬性。

Copyright ? 2019- 91gzw.com 版權(quán)所有 湘ICP備2023023988號-2

違法及侵權(quán)請聯(lián)系:TEL:199 1889 7713 E-MAIL:2724546146@qq.com

本站由北京市萬商天勤律師事務(wù)所王興未律師提供法律服務(wù)