日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
深入理解JavaScript作用域和作用域鏈

【稿件】

創(chuàng)新互聯(lián)主要為客戶提供服務(wù)項(xiàng)目涵蓋了網(wǎng)頁(yè)視覺設(shè)計(jì)、VI標(biāo)志設(shè)計(jì)、營(yíng)銷型網(wǎng)站建設(shè)、網(wǎng)站程序開發(fā)、HTML5響應(yīng)式網(wǎng)站建設(shè)公司、手機(jī)網(wǎng)站開發(fā)、微商城、網(wǎng)站托管及成都企業(yè)網(wǎng)站維護(hù)、WEB系統(tǒng)開發(fā)、域名注冊(cè)、國(guó)內(nèi)外服務(wù)器租用、視頻、平面設(shè)計(jì)、SEO優(yōu)化排名。設(shè)計(jì)、前端、后端三個(gè)建站步驟的完善服務(wù)體系。一人跟蹤測(cè)試的建站服務(wù)標(biāo)準(zhǔn)。已經(jīng)為成都純水機(jī)行業(yè)客戶提供了網(wǎng)站制作服務(wù)。

前言

JavaScript中有一個(gè)被稱為作用域(Scope)的特性。雖然對(duì)于許多新手開發(fā)者來說,作用域的概念并不是很容易理解,本文我會(huì)盡我所能用最簡(jiǎn)單的方式來解釋作用域和作用域鏈,希望大家有所收獲!

作用域(Scope)

1.什么是作用域

作用域是在運(yùn)行時(shí)代碼中的某些特定部分中變量,函數(shù)和對(duì)象的可訪問性。換句話說,作用域決定了代碼區(qū)塊中變量和其他資源的可見性??赡苓@兩句話并不好理解,我們先來看個(gè)例子:

 
 
 
 
  1. function outFun2() { 
  2.   var inVariable = "內(nèi)層變量2"; 
  3. outFun2();//要先執(zhí)行這個(gè)函數(shù),否則根本不知道里面是啥 
  4. console.log(inVariable); // Uncaught ReferenceError: inVariable is not defined 

從上面的例子可以體會(huì)到作用域的概念,變量inVariable在全局作用域沒有聲明,所以在全局作用域下取值會(huì)報(bào)錯(cuò)。我們可以這樣理解:作用域就是一個(gè)獨(dú)立的地盤,讓變量不會(huì)外泄、暴露出去。也就是說作用域***的用處就是隔離變量,不同作用域下同名變量不會(huì)有沖突。

ES6 之前 JavaScript 沒有塊級(jí)作用域,只有全局作用域和函數(shù)作用域。ES6的到來,為我們提供了‘塊級(jí)作用域’,可通過新增命令let和const來體現(xiàn)。

2.全局作用域和函數(shù)作用域

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

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

 
 
 
 
  1. var outVariable = "我是最外層變量"; //最外層變量 
  2. function outFun() { //最外層函數(shù) 
  3.   var inVariable = "內(nèi)層變量"; 
  4.   function innerFun() { //內(nèi)層函數(shù) 
  5.       console.log(inVariable); 
  6.   } 
  7.   innerFun(); 
  8. console.log(outVariable); //我是最外層變量 
  9. outFun(); //內(nèi)層變量 
  10. console.log(inVariable); //inVariable is not defined 
  11. innerFun(); //innerFun is not defined 
  • 所有未定義直接賦值的變量自動(dòng)聲明為擁有全局作用域

 
 
 
 
  1. function outFun2() { 
  2.   variable = "未定義直接賦值的變量"; 
  3.   var inVariable2 = "內(nèi)層變量2"; 
  4. outFun2();//要先執(zhí)行這個(gè)函數(shù),否則根本不知道里面是啥 
  5. console.log(variable); //未定義直接賦值的變量 
  6. console.log(inVariable2); //inVariable2 is not defined 
  • 所有window對(duì)象的屬性擁有全局作用域

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

全局作用域有個(gè)弊端:如果我們寫了很多行 JS 代碼,變量定義都沒有用函數(shù)包括,那么它們就全部都在全局作用域中。這樣就會(huì)污染全局命名空間,容易引起命名沖突。

 
 
 
 
  1. // 張三寫的代碼中 
  2. var data = {a: 100} 
  3.  
  4. // 李四寫的代碼中 
  5. var data = {x: true} 

這就是為何 jQuery、Zepto 等庫(kù)的源碼,所有的代碼都會(huì)放在(function(){....})()中。因?yàn)榉旁诶锩娴乃凶兞?,都不?huì)被外泄和暴露,不會(huì)污染到外面,不會(huì)對(duì)其他的庫(kù)或者 JS 腳本造成影響。這是函數(shù)作用域的一個(gè)體現(xiàn)。

函數(shù)作用域,是指聲明在函數(shù)內(nèi)部的變量,和全局作用域相反,局部作用域一般只在固定的代碼片段內(nèi)可訪問到,最常見的例如函數(shù)內(nèi)部。

 
 
 
 
  1. function doSomething(){ 
  2.   var blogName="浪里行舟"; 
  3.   function innerSay(){ 
  4.       alert(blogName); 
  5.   } 
  6.   innerSay(); 
  7. alert(blogName); //腳本錯(cuò)誤 
  8. innerSay(); //腳本錯(cuò)誤 

作用域是分層的,內(nèi)層作用域可以訪問外層作用域的變量,反之則不行。我們看個(gè)例子,用泡泡來比喻作用域可能好理解一點(diǎn):

***輸出的結(jié)果為 2, 4, 12

  • 泡泡1是全局作用域,有標(biāo)識(shí)符foo;

  • 泡泡2是作用域foo,有標(biāo)識(shí)符a,bar,b;

  • 泡泡3是作用域bar,僅有標(biāo)識(shí)符c。

3.塊級(jí)作用域

ES6 中開始加入了塊級(jí)作用域,可通過新增命令let和const來體現(xiàn),如下:

 
 
 
 
  1. if (true) { 
  2.   let name = 'zhangsan' 
  3. console.log(name) // 報(bào)錯(cuò),因?yàn)閘et定義的name是在if這個(gè)塊級(jí)作用域 

值得注意的是:塊語(yǔ)句(大括號(hào)“{}”中間的語(yǔ)句),如 if 和 switch 條件語(yǔ)句或 for 和 while 循環(huán)語(yǔ)句,不像函數(shù),它們不會(huì)創(chuàng)建一個(gè)新的作用域。在塊語(yǔ)句中定義的變量將保留在它們已經(jīng)存在的作用域中。

 
 
 
 
  1. if (true) { 
  2.   // 'if' 條件語(yǔ)句塊不會(huì)創(chuàng)建一個(gè)新的作用域 
  3.   var name = 'Hammad'; // name 依然在全局作用域中 
  4. console.log(name); // logs 'Hammad' 

作用域鏈

1.什么是自由變量

首先認(rèn)識(shí)一下什么叫做 自由變量 。如下代碼中,console.log(a)要得到a變量,但是在當(dāng)前的作用域中沒有定義a(可對(duì)比一下b)。當(dāng)前作用域沒有定義的變量,這成為自由變量 。自由變量的值如何得到 —— 向父級(jí)作用域?qū)ふ遥ㄗ⒁猓哼@種說法并不嚴(yán)謹(jǐn),下文會(huì)重點(diǎn)解釋)。

 
 
 
 
  1. var a = 100 
  2. function fn() { 
  3.   var b = 200 
  4.   console.log(a) // 這里的a在這里就是一個(gè)自由變量 
  5.   console.log(b) 
  6. fn() 

2.什么是作用域鏈

如果父級(jí)也沒呢?再一層一層向上尋找,直到找到全局作用域還是沒找到,就宣布放棄。這種一層一層的關(guān)系,就是作用域鏈 。

 
 
 
 
  1. var a = 100 
  2. function F1() { 
  3.   var b = 200 
  4.   function F2() { 
  5.       var c = 300 
  6.       console.log(a) // 自由變量,順作用域鏈向父作用域找 
  7.       console.log(b) // 自由變量,順作用域鏈向父作用域找 
  8.       console.log(c) // 本作用域的變量 
  9.   } 
  10.   F2() 
  11. F1() 

3.關(guān)于自由變量的取值

關(guān)于自由變量的值,上文提到要到父作用域中取,其實(shí)有時(shí)候這種解釋會(huì)產(chǎn)生歧義。

 
 
 
 
  1. var x = 10 
  2. function fn() { 
  3. console.log(x) 
  4. function show(f) { 
  5. var x = 20 
  6. (function() { 
  7.   f() //10,而不是20 
  8. })() 
  9. show(fn) 

在fn函數(shù)中,取自由變量x的值時(shí),要到哪個(gè)作用域中取?——要到創(chuàng)建fn函數(shù)的那個(gè)作用域中取,無(wú)論fn函數(shù)將在哪里調(diào)用。

所以,不要在用以上說法了。相比而言,用這句話描述會(huì)更加貼切:要到創(chuàng)建這個(gè)函數(shù)的那個(gè)域”。 作用域中取值,這里強(qiáng)調(diào)的是“創(chuàng)建”,而不是“調(diào)用”,切記切記——其實(shí)這就是所謂的"靜態(tài)作用域"。

 
 
 
 
  1. var a = 10 
  2. function fn() { 
  3. var b = 20 
  4. function bar() { 
  5.   console.log(a + b) //30 
  6. return bar 
  7. var x = fn(), 
  8. b = 200 
  9. x() //bar() 

fn()返回的是bar函數(shù),賦值給x。執(zhí)行x(),即執(zhí)行bar函數(shù)代碼。取b的值時(shí),直接在fn作用域取出。取a的值時(shí),試圖在fn作用域取,但是取不到,只能轉(zhuǎn)向創(chuàng)建fn的那個(gè)作用域中去查找,結(jié)果找到了,所以***的結(jié)果是30。

作用域與執(zhí)行上下文

許多開發(fā)人員經(jīng)?;煜饔糜蚝蛨?zhí)行上下文的概念,誤認(rèn)為它們是相同的概念,但事實(shí)并非如此。

我們知道JavaScript屬于解釋型語(yǔ)言,JavaScript的執(zhí)行分為:解釋和執(zhí)行兩個(gè)階段。

解釋階段:

  • 詞法分析

  • 語(yǔ)法分析

  • 作用域規(guī)則確定

執(zhí)行階段:

  • 創(chuàng)建執(zhí)行上下文

  • 執(zhí)行函數(shù)代碼

  • 垃圾回收

上文我們提到作用域只是一個(gè)“地盤”,一個(gè)抽象的概念,其中沒有變量。要通過作用域?qū)?yīng)的執(zhí)行上下文環(huán)境來獲取變量的值。一個(gè)作用域下可能包含若干個(gè)上下文環(huán)境。有可能從來沒有過上下文環(huán)境(函數(shù)從來就沒有被調(diào)用過);有可能有過,現(xiàn)在函數(shù)被調(diào)用完畢后,上下文環(huán)境被銷毀了;有可能同時(shí)存在一個(gè)或多個(gè)(閉包)。同一個(gè)作用域下,不同的調(diào)用會(huì)產(chǎn)生不同的執(zhí)行上下文環(huán)境,繼而產(chǎn)生不同的變量的值。

JavaScript解釋階段便會(huì)確定作用域規(guī)則,因此作用域在函數(shù)定義時(shí)就已經(jīng)確定了,而不是在函數(shù)調(diào)用時(shí)確定。

作用域和執(zhí)行上下文之間***的區(qū)別是: 執(zhí)行上下文在運(yùn)行時(shí)確定,隨時(shí)可能改變;作用域在定義時(shí)就確定,并且不會(huì)改變。

 
 
 
 
  1. var a=1;     //全局作用域 
  2. function fn1(){ 
  3.   var a=2; //fn1作用域 

如上述代碼,作用域代表著已聲明變量或者函數(shù)的訪問范圍,在fn1作用域內(nèi)使用變量a會(huì)先從當(dāng)前作用域?qū)ふ?,如果沒有會(huì)往上一層作用域?qū)ふ摇?/p>

 
 
 
 
  1. this.a=1;     //全局執(zhí)行上下文 
  2. function fn1(){ 
  3. this.a=2;   //fn1執(zhí)行上下文 
  4. var obj=new fn1(); 

如上述代碼,new fn1()時(shí)的執(zhí)行上下文就是obj 可以看出,作用域和執(zhí)行上下文并不是一個(gè)概念,它們的區(qū)別就像是你訪問a和this.a時(shí)的區(qū)別。

參考文章

  • 深入理解javascript原型和閉包系列

  • Web 前端面試指南與高頻考題解析

  • 深入理解JS中聲明提升、作用域(鏈)和this關(guān)鍵字

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

  • JavaScript 作用域和作用域鏈

  • javascript作用域和執(zhí)行上下文的區(qū)別

浪里行舟,碩士研究生,專注于前端。個(gè)人公眾號(hào):「前端工匠」,致力于打造適合初中級(jí)工程師能夠快速吸收的一系列優(yōu)質(zhì)文章!

【原創(chuàng)稿件,合作站點(diǎn)轉(zhuǎn)載請(qǐng)注明原文作者和出處為.com】


文章名稱:深入理解JavaScript作用域和作用域鏈
文章轉(zhuǎn)載:http://www.dlmjj.cn/article/cdjgccc.html