js作用域鏈

純干貨,沒有廢話

先看題目杏节,諸位自測

題目1

var a = 1
function fn1(){
  function fn2(){
    console.log(a)
  }
  function fn3(){
    var a = 4
    fn2()
  }
  var a = 2
  return fn3
}
var fn = fn1()
fn() //輸出多少

題目2

var a = 1
function fn1(){
  function fn3(){
    var a = 4
    fn2()
  }
  var a = 2
  return fn3
}
function fn2(){
  console.log(a)
}
var fn = fn1()
fn() //輸出多少

題目3

var a = 1
function fn1(){

  function fn3(){
    function fn2(){
      console.log(a)
    }
    fn2()
    var a = 4
  }
  var a = 2
  return fn3
}
var fn = fn1()
fn() //輸出多少

結(jié)果

2//題目1
1//題目2
undefined//題目3

解密

1、函數(shù)在執(zhí)行時先從自己內(nèi)部的局部作用域中查找變量,找到則用
2奋渔、如果在函數(shù)內(nèi)部找不到镊逝,則在定義此函數(shù)的作用域中查找變量,找到則用
3嫉鲸、如果在定于函數(shù)的作用域內(nèi)還是找不到撑蒜,則在定義包含此函數(shù)的函數(shù)的作用域中查找變量,找到則用
4玄渗、依次往上座菠,直至全局作用域,如果在全局作用域中依然找不到捻爷,則報錯

結(jié)果淺析

題目1:

執(zhí)行fn()即執(zhí)行fn1返回的fn3辈灼,在fn3中又調(diào)用fn2,fn2中用到變量a也榄,根據(jù)作用域鏈規(guī)則查找巡莹,首先在fn2內(nèi)部沒有找到變量a,則繼續(xù)在定義fn2的作用域(題目1中即為函數(shù)fn1的內(nèi)部)中查找變量a甜紫,在fn1中定于了變量a=2降宅,因此執(zhí)行fn()得到結(jié)果為2

題目2與題目1類似,不在贅述

題目3:

按照題目1的思路囚霸,在題目三中腰根,在定義fn2的作用域中也有變量a=4,那為什么執(zhí)行fn()得到的結(jié)果為undefined呢拓型?那就要引入一個js中變量聲明前置的知識點
什么是變量聲明前置呢额嘿?
所謂變量聲明前置就是js引擎在解析js代碼時,會將所有的變量都提升至所有js代碼的最前面來聲明(注意:僅聲明提前劣挫,賦值不提前)
看個例子:

console.log(a)
var a = 1
//打印結(jié)果是什么?
//A.1
//B.報錯:a未定義
//C.undefined

答案是undefined
因為存在變量聲明前置册养,所以上面例子的代碼等價于:

var a ;//undefined
console.log(a) ;//執(zhí)行此語句時,變量a已經(jīng)聲明压固,但未賦值球拦,因此值為undefined
a = 1

回到題目3:
結(jié)合變量聲明前置,函數(shù)fn3等同于:

function fn3() {
  function fn2() {
    console.log(a)
  }
  var a
  fn2()//執(zhí)行fn2()時帐我,變量a已經(jīng)聲明坎炼,但未賦值,因此值為undefined
  a = 4
}

簡單理解作用域鏈:
先找自己拦键,自己沒有找爹谣光,爹沒有找爹的爹,爹的爹沒有找爹的爹的爹芬为,......抢肛,直至找到?jīng)]有爹可找

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末狼钮,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子捡絮,更是在濱河造成了極大的恐慌熬芜,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件福稳,死亡現(xiàn)場離奇詭異涎拉,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)的圆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進(jìn)店門鼓拧,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人越妈,你說我怎么就攤上這事季俩。” “怎么了梅掠?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵酌住,是天一觀的道長。 經(jīng)常有香客問我阎抒,道長酪我,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任且叁,我火速辦了婚禮都哭,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘逞带。我一直安慰自己欺矫,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布展氓。 她就那樣靜靜地躺著穆趴,像睡著了一般。 火紅的嫁衣襯著肌膚如雪带饱。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天阅羹,我揣著相機(jī)與錄音勺疼,去河邊找鬼。 笑死捏鱼,一個胖子當(dāng)著我的面吹牛执庐,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播导梆,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼轨淌,長吁一口氣:“原來是場噩夢啊……” “哼迂烁!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起递鹉,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤盟步,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后躏结,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體却盘,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年媳拴,在試婚紗的時候發(fā)現(xiàn)自己被綠了黄橘。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,711評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡屈溉,死狀恐怖塞关,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情子巾,我是刑警寧澤帆赢,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站砰左,受9級特大地震影響匿醒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜缠导,卻給世界環(huán)境...
    茶點故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一廉羔、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧僻造,春花似錦憋他、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至立膛,卻和暖如春揪罕,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背宝泵。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工好啰, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人儿奶。 一個月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓框往,卻偏偏與公主長得像,于是被迫代替她去往敵國和親闯捎。 傳聞我的和親對象是個殘疾皇子椰弊,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,611評論 2 353

推薦閱讀更多精彩內(nèi)容

  • 例題: 例題中的作用域鏈: console.log('1.'+a):在JS中的語句var a = 1其實是兩條語句...
    老虎愛吃母雞閱讀 534評論 1 3
  • 給出兩個不同的例子: 范例1: 分析上面代碼:全局作用域中定義了變量x,function foo()以及funct...
    DeeJay_Y閱讀 342評論 0 1
  • 函數(shù)聲明和函數(shù)表達(dá)式有什么區(qū)別许溅? 1.聲明的函數(shù)foo將會在執(zhí)行前提升,因此foo在函數(shù)上下文都是可以被調(diào)用的秉版,即...
    小囧兔閱讀 300評論 0 1
  • 之前寫過一篇JavaScript 閉包究竟是什么的文章理解閉包贤重,覺得寫得很清晰,可以簡單理解閉包產(chǎn)生原因沐飘,但看評論...
    寧驥閱讀 335評論 0 1
  • 大家好耐朴,我是IT修真院成都分院第三期的學(xué)員借卧,一枚正直純潔善良的web程序員今天給大家分享一下,修真院官網(wǎng)js任務(wù)4...
    取老夫的草莓蛋糕來閱讀 229評論 0 2