我第一次真正理解了作用域链

最近在摸鱼的时候忽然刷到一个面试题,考察的是有关作用域链的,当时就想八股文里应该没有比作用域链更简单的了,后来果然毫无意外的答错了,原来我根本就不了解它,只是以前没有撞枪口上,这下栽了。本文就浅浅记录一下我更正错误的决心

面试题

js 复制代码
function bar() {
    console.log(myName)
}
function foo() {
    var myName = "foo里面"
    bar()
}
var myName = "foo外面"
foo()

看到题目的第一反应,就判断了答案是foo里面,想着bar里面没有myName,然后直接找他的上级foo,答案不就出来了吗,控制台直接打印,GG......

调用栈

首先,要理解作用域链要先理解好js的执行机制,抛开变量提升什么乱七八糟的先不谈,js的执行机制概括来说就是先编译,后执行。我们主要分析的是编译阶段。

一段js代码在执行前,都会编译并创建执行上下文,一般有以下几种情况

  • JavaScript 执行全局代码的时候,会编译全局代码并创建全局执行上下文,而且在整个页面的生存周期内,全局执行上下文只有一份。
  • 当调用一个函数的时候,函数体内的代码会被编译,并创建函数执行上下文,一般情况下,函数执行结束之后,创建的函数执行上下文会被销毁。
  • 当使用 eval 函数的时候,eval 的代码也会被编译,并创建执行上下文。

编译阶段创建好执行上下文,压入调用栈,遵循先进后出的原则

由此,根据代码,可以得到以下调用栈的示意图

let、const关键字定义的变量不存在变量提升,才会被保存在词法环境

按照上面的图,我坚信我自己是对的,肯定是js有毒

outer

经过进一步查资料,发现了它 - outer

其实在每个执行上下文的变量环境中,都包含了一个外部引用,用来指向外部的执行上下文,我们把这个外部引用称为 outer

词法作用域

可是我们不知道outer的指向规则是什么,主观意识就觉得会指向foo的 执行上下文,但其实不然,还和另一个概念有关,它叫做词法作用域

词法作用域就是指作用域是由代码中函数声明位置来决定的,所以词法作用域是静态的作用域,通过它就能够预测代码在执行过程中如何查找标识符

作用域链是由词法作用域决定的

真相

引入新概念,我们可以把上图优化一下

从图中可以看出,barfooouter 都是指向全局上下文的,这也就意味着如果在 bar或者 foo中使用了外部变量,那么 JavaScript 引擎会去全局执行上下文中查找。我们把这个查找的链条就称为作用域链。同时也说明了,词法作用域是代码编译阶段就决定好的,和函数是怎么调用的没有关系。

相关推荐
ZhengEnCi3 分钟前
Q04-Vite禁用CSS代码分割-解决生产环境样式加载顺序混乱问题
前端·vue.js·vite
九酒34 分钟前
AI Agent 开发踩坑记:口播功能非得用 APP 原生实现吗?
前端·人工智能·agent
Jackson__1 小时前
做了一段时间的AI coding后,我终于搞清了 CLI 和 MCP 的区别
前端·agent·ai编程
IT_陈寒4 小时前
JavaScript项目实战经验分享
前端·人工智能·后端
用户47949283569154 小时前
6w star,GitHub 趋势第一的 Ponytail,这个agent插件到底在火什么
前端·后端
薛定喵的谔6 小时前
我开源了一个精致的 Next.js 博客模板:Skyplume
前端·前端框架·next.js
张龙6876 小时前
构建生产级 AI Agent:工具调用与记忆架构实战指南
前端
kyriewen7 小时前
2026 年了,还在用 Node.js?Bun 迁移实战:20 分钟搞定,附踩坑记录
前端·javascript·node.js
青山Coding9 小时前
Cesium应用(八):物体运动的实现思路
前端·cesium
用户41659673693559 小时前
Android WebView 加载 file:// 离线页面调试教程
android·前端