JS执行上下文和变量提升

执行上下文(execution context)是 JavaScript 中的一个基本部分,可以大致理解为当前被执行代码的环境或者作用域。变量提升(hoisting)也是一个基本概念,简单地说就是,我可以在变量和函数声明之前就使用它们,其中具体的原因又和执行上下文有着密切的关系。

首先,请看一个例子:

(function() {
  console.log(typeof team);
  console.log(typeof player);
  var team = 'lakers',
      player = function() {
        return 'kobe bryant';
      };
  function team() {
    return 'bulls';
  }
  console.log(typeof team);
  console.log(typeof player);
}());

试想一下这几个 console.log 会分别打印出来什么。

这里的答案是:

// function
// undefined
// string
// function

如果和你的答案不一致,那么,请看下文。

要弄清楚原因,首先要明白以下几点内容:

第一点

JavaScript 代码执行时的执行环境有以下几种:

  • 全局环境
  • 函数环境
  • eval

其中,全局环境为 JavaScript 代码执行时首次进入的默认环境;函数环境为每当一个函数被调用时,会进入该函数内部的一个环境;eval 内部的文本被执行时也会相应地产生一个执行环境(eval 不被推荐使用)。

第二点

浏览器里的 JavaScript 解释器是单线程的,同一时间里只能做一件事情。代码执行时,会产生并进入不同的执行上下文,这些执行上下文会构成一个执行上下文栈(execution context stack),这个栈的栈底永远是全局上下文,栈顶是当前正在执行的上下文。代码执行过程中,每生成一个执行上下文,都会压入执行上下文栈中,当栈顶的上下文执行完毕后,会自动出栈,控制权给到当前的栈。

一个例子:

function foo() {
  console.log('我是foo里面的代码')
  function bar() {
    console.log('我是bar里面的代码')
    function baz() {
      console.log('我是baz里面的代码')
    }
    baz()
  }
  function box() {
    console.log('我是box里面的代码')
  }
  bar()
  box()
}
foo()

上下文的出入栈过程在该例中的表现为:

  1. 全局执行上下文入栈。

  2. 执行 foo(),创建 foo 对应的执行上下文,入栈。

  3. 接着执行 foo 内部的代码,到了 bar(),创建 bar 的上下文,入栈。

  4. 继续执行 bar 内部的代码,到了 baz(),创建 baz 的上下文,入栈。

  5. baz 里没有再创建新的上下文,代码执行完毕之后,baz 的执行上下文出栈。

  6. baz 的执行上下文出栈后,bar 里面也没有其它可执行代码,bar 的执行上下文出栈。

  7. bar 的执行上下文出栈后,继续执行 foo 内部剩余可执行代码,即 box(),随即创建 box 的执行上下文,入栈。

  8. box 内部没有再创建新的上下文,代码执行完毕之后,出栈。

这时,就只剩全局执行上下文了。

可以用 console.trace() 来验证一下:

function foo() {
  console.log('我是foo里面的代码')
  console.trace()
  function bar() {
    console.log('我是bar里面的代码')
    console.trace()
    function baz() {
      console.log('我是baz里面的代码')
      console.trace()
    }
    baz()
  }
  function box() {
    console.log('我是box里面的代码')
    console.trace()
  }
  bar()
  box()
}
console.trace()
foo()
console.trace()

结果和预期是一致的:

stack.png

总结,关于调用栈,有几个关键点:

  • 单线程。

  • 同步执行,所有的执行上下文都要等到栈顶的上下文出栈以后才能依次执行。

  • 唯一的全局上下文。

  • 无限个数的函数上下文。

  • 每一次函数被调用时都会创建新的执行上下文,包括调用自己。

第三点

在 JS 解释器内部,每次调用执行上下文分为两个阶段:

  1. 创建阶段

这个阶段实际上处于函数被调用,但还未真正执行其内部的代码。

该阶段主要做了三件事情:

  • 创建作用域链。

  • 创建包含变量、函数和参数的变量对象。

    具体来说,首先,根据函数的参数创建 arguments 对象,初始化参数名称和值。然后,扫描上下文的函数声明,将函数名称作为属性名存入变量对象中,并且该属性指向函数在内存中的引用地址,如果函数名称已经存在,那么引用的指针将被重写。最后,扫描上下文的变量声明,将变量名称作为属性名存入变量对象中,并且将属性值初始化为 undefined,但是,如果变量名称已经存在于变量对象中,则对当前变量不进行任何操作,继续扫描。

    值得注意的是,解释器先扫描的是函数声明,然后才是变量声明,这和代码的书写顺序无关。

  • 设置当前上下文 this 的值。

那么,执行上下文便可以抽象为一个对象:

executionContextObj = {
  scopeChain: {...},
  variableObject: {...},
  this: {...}
}
  1. 激活/代码执行阶段

这个阶段会逐行运行代码,过程中,设置变量的值和函数的引用,解释/执行代码。

为了进一步了解这两个阶段,可以举个例子:

function player(playerTeam) {
  var player = 'kobe bryant'
  var getPlayer = function getYourPlayer() {
    return `You got ${player}!`
  }
  function getTeam() {
    return playerTeam
  }
}
player('lakers')

当调用 player('lakers') 时,进入创建阶段,创建的执行上下文抽象成对象以后应该是这样子:

playerExecutionContext = {
  scopeChain: { ... },
  variableObject: {
    arguments: {
      0: 'lakers',
      length: 1
    },
    playerTeam: 'lakers',
    getTeam: pointer to function getTeam() {...}, /* 保存着一个指向函数 getTeam 的指针*/
    player: undefined,
    getPlayer: undefined
  },
  this: { ... }
}

之后进入激活/代码执行阶段,执行完之后,抽象出来的对象会变成这个样子:

playerExecutionContext = {
  scopeChain: { ... },
  variableObject: {
    arguments: {
      0: 'lakers',
      length: 1
    },
    playerTeam: 'lakers',
    getTeam: pointer to function getTeam() {...}, // 保存着一个指向函数 getTeam 的指针
    player: 'kobe bryant',
    getPlayer: pointer to function getYourPlayer() {...} // 保存着一个指向函数 getYourPlayer 的指针
  },
  this: { ... }
}

了解了这些以后,文章一开始的那个例子就明朗了起来:

(function() {
  console.log(typeof team); // function
  console.log(typeof player); // undefined
  var team = 'lakers',
      player = function() {
        return 'kobe bryant';
      };
  function team() {
    return 'bulls';
  }
  console.log(typeof team); // string
  console.log(typeof player); // function
}());

这是一个立即执行函数,执行上下文会被马上创建出来,代码中,前两个 console.log 执行时,其它代码还没有真正执行,那其实可以理解为前两个 console.log 处于创建阶段之后,以及激活/代码执行阶段之前,而后两个 console.log 处于激活/代码执行阶段之后。

创建阶段,解释器先扫描函数声明,变量对象中就有了 team,指向 team 函数在内存中的引用地址。然后扫描变量声明,遇到变量 team,发现同名属性已经存在于变量对象中,跳过,继续扫描,遇到变量 player,将其存入变量对象中,并赋值 undefined

如此一来,第一个 log 打印出来的就是 function,第二个 log 打印出来的是 undefined

激活/代码执行阶段,将 team 赋值为 'lakers',将 player 赋值为函数表达式。

那么,第三个 log 打印出来的就是 string,第四个 log 打印出来的为 function

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容