深入学习JavaScript系列——作用域和作用域链

简介: 深入学习JavaScript系列——作用域和作用域链

大厂面试题分享 面试题库

前后端面试题库 (面试必备) 推荐:★★★★★

地址:前端面试题库

1、js执行上下文和作用域链的关系?

作用域链和执行上下文是紧密相连的概念。

执行上下文是JavaScript代码执行时的环境,包括变量、函数、参数等信息。每个函数都有自己的执行上下文,而全局代码也有自己的执行上下文。

从执行上下文的角度来说: 作用域链是由当前执行上下文和所有外层执行上下文的变量对象组成的链式结构。当JavaScript代码在一个执行上下文中查找变量时,会先在当前执行上下文的变量对象中查找,如果没有找到,就会继续在外层执行上下文的变量对象中查找,直到找到该变量或者到达全局执行上下文。

因此,作用域链的形成是由执行上下文的嵌套关系决定的。在函数定义时,就已经确定了该函数的作用域链。当函数被调用时,会创建一个新的执行上下文,并将该执行上下文的变量对象添加到作用域链的顶端,从而形成新的作用域链。 函数销毁时,与之对应的作用域链节点也会销毁。

总之,作用域链和执行上下文是密不可分的,它们共同构成了JavaScript代码的作用域和变量查找机制。

2、什么是js作用域链和作用域?

在 JavaScript 中,每个函数都有一个作用域链,它是一个由当前函数和所有外层函数的变量对象组成的列表。当 JavaScript 引擎查找一个变量时,它会先在当前函数的变量对象中查找,如果找不到,就会在外层函数的变量对象中查找,直到找到该变量或者到达全局对象为止。

每个函数内部声明的变量和函数都只能在该函数内部访问,外部无法访问。但是,如果一个函数内部嵌套了另一个函数,那么内部函数可以访问外部函数的变量和函数,这就是作用域链的作用。

作用域链的创建是在函数定义时确定的,而不是在函数调用时确定的。当一个函数被调用时,它会创建一个新的执行上下文,并将其添加到调用栈中。在执行上下文中,会创建一个变量对象,该变量对象包含了当前函数的所有变量和函数声明。同时,该执行上下文的作用域链会指向当前函数的作用域链。

当 JavaScript 引擎在执行一个函数时,如果需要访问一个变量,它会先在当前函数的变量对象中查找,如果找不到,就会在外层函数的变量对象中查找,直到找到该变量或者到达全局对象为止。这个查找的过程就是作用域链的遍历过程。

通过上面两个问题的展开,其实已经能大致了解了作用域与作用域链,下面就展开讲一讲 在学习作用域链时需要注意哪些内容

3、作用域和作用域链的创建

js采用的是静态作用域,也就是说js函数的作用域是在函数定义的情况下就已经确定了,那么作用域链也是在函数定义的时候就创建,

举个静态作用域的小例子:

var value = 1;
function foo() {
    console.log(value);
}
function bar() {
    var value = 2;
    foo();
}
bar();
// 结果是 1 在函数定义的时候确定的作用value等于1 所以调用的时候也等于1 复制代码

如果上述的代码是动态作用域,那么value就是在执行时才生成 所以打印为2

动态作用域的语言:

  1. Bash:Bash 是一种常用的 Unix shell,它支持动态作用域。在 Bash 中,变量的作用域是在函数被调用时
  2. Perl:Perl 是一种通用的高级编程语言,它支持动态作用域。在 Perl 中,变量的作用域是在程序运行时确定
  3. Lisp:Lisp 是一种函数式编程语言,它也支持动态作用域。在 Lisp 中,变量的作用域是在函数被调用时确定
  4. Emacs Lisp:Emacs Lisp 是一种基于 Lisp 的编程语言,它是 Emacs 编辑器的扩展语言,也支持动态作用域。

4 作用域和作用域链的执行销毁

js有两种函数作用域:全局作用域和函数作用域,创建上面已经讲到了,那么执行呢,其实是在函数中的变量被使用时会执行,先在当前作用域中查找变量,然后再逐步往上查找父级作用域,直到找到为止, 如果最终都没找到,那么就会抛出一个 ReferenceError 异常。

作用域链及作用域链的销毁

当一个函数执行完毕之后,那么它的执行上下文也随之销毁,于此同时。对应的函数作用域和作用域链也会销毁(当前函数作用域节点),函数中声明的变量也会被销毁,这就是js中大垃圾回收机制(gc) 最常见的有标记清除法和计数算法

注:如果函数中使用了闭包,那可能包含外部变量,此时只有外部变量被销毁时才能销毁对应的闭包。这就是所谓的内存泄漏。

5 几个作用域练习题目

题目一:

var scope= "global scope";
function checkscope(){
    var scope= "local scope";
    functionscope(){
        returnscope;
    }
    returnscope();
}
checkscope();
//localscope复制代码
var scope= "global scope";
function checkscope(){
    var scope= "local scope";
    functionscope(){
        returnscope;
    }
    returnscope;
}
checkscope()();
//localscope复制代码

解释:函数的作用域基于函数创建的位置,

注:checkscope()()和checkscope()的区别:

题目一中,checkscope()是一个函数,它返回另一个函数scope()。这个内部函数scope()可以访问checkscope()函数的局部变量。它是一个闭包函数,因为它可以访问其外部函数checkscope()的词法环境。

checkscope()()是在调用checkscope()返回的函数scope()。它不是直接调用checkscope()函数本身。由于checkscope()函数返回了一个函数,因此需要在checkscope()后添加另一个括号来调用内部函数scope()。相当于》checkscope(). f()

题目二:

var a = 1;
function foo() {
  console.log(a);
  var a = 2;
}
foo();
// undefined
复制代码

解释:因为在函数 foo() 中,先执行了 var a = 2 语句,此时变量 a 被重新定义并赋值为 2。所以,在函数 foo() 中,变量 a 的值是 2,而不是全局变量 a 的值。因此,执行 console.log(a) 语句时,输出的是 undefined,因为变量 a 被重新定义,但没有被初始化。 变量能提升 但是赋值不能提升

题目三:

var x = 1;
function outer() {
  var y = 2;
  function inner() {
    var z = 3;
    console.log(x + y + z);
  }
  inner();
}
outer();
// 6
复制代码

解释:在调用inner函数时,可以访问外层作用域中的变量 x 和 y。变量 x 的值是 1,变量 y 的值是 2,变量 z 的值是 3。所以,执行 console.log(x + y + z) 语句时,输出的是 6。

题目四:

function foo() {
  var a = 1;
  function bar() {
    console.log(a);
  }
  return bar;
}
var baz = foo();
baz();复制代码

答案:代码输出的是1。因为bar函数在定义时可以访问到其外层函数foo的变量a,并将其保存在函数作用域内部。当执行baz()时,实际上是执行了内部函数bar(),此时访问到的变量a是定义时保存在bar函数作用域内的变量,所以输出的是1

题目五:

var a = 1;
function foo() {
  console.log(a);
}
function bar() {
  var a = 2;
  foo();
}
bar();复制代码

答案:代码输出的是1。因为bar函数定义了一个名为a的局部变量,并将其赋值为2,但是在调用foo函数时,它会在全局作用域中查找变量a,因为在foo函数内部没有定义变量a。因此,foo函数输出的是全局变量a的值,即1。

关键点 foo的父级作用域不是bar 而是全局 只是foo()在bar()中执行。

题目六

var a = 1;
function foo() {
  var a = 2;
  function bar() {
    console.log(a);
  }
  return bar;
}
var baz = foo();
baz();复制代码

答案:代码输出的是2。在foo函数内部定义了一个局部变量a,并将其赋值为2,然后返回了内部定义的函数bar。bar函数可以访问到foo函数的作用域链,因此可以访问到变量a。当调用baz函数时,实际上是执行了内部函数bar,此时访问到的变量a是定义时保存在foo函数作用域内的变量,所以输出的是2。

和题目五对比,因为bar函数是定义在foo函数内的 ,所以bar的父级作用域是foo

看到这里的同学都很厉害,那就顺便给我点个赞吧!

大厂面试题分享 面试题库

前后端面试题库 (面试必备) 推荐:★★★★★

地址:前端面试题库

相关文章
|
2月前
|
JavaScript 前端开发
js的作用域作用域链
【10月更文挑战第29天】理解JavaScript的作用域和作用域链对于正确理解变量的访问和生命周期、避免变量命名冲突以及编写高质量的JavaScript代码都具有重要意义。在实际开发中,需要合理地利用作用域和作用域链来组织代码结构,提高代码的可读性和可维护性。
|
2月前
|
自然语言处理 JavaScript 前端开发
[JS]作用域的“生产者”——词法作用域
本文介绍了JavaScript中的作用域模型与作用域,包括词法作用域和动态作用域的区别,以及全局作用域、函数作用域和块级作用域的特点。通过具体示例详细解析了变量提升、块级作用域中的暂时性死区等问题,并探讨了如何在循环中使用`var`和`let`的不同效果。最后,介绍了两种可以“欺骗”词法作用域的方法:`eval(str)`和`with(obj)`。文章结合了多位博主的总结,帮助读者更快速、便捷地掌握这些知识点。
37 2
[JS]作用域的“生产者”——词法作用域
|
2月前
|
Web App开发 JavaScript 前端开发
如何学习JavaScript?
如何学习JavaScript?
55 5
|
2月前
|
JavaScript 前端开发 索引
JavaScript学习第二章--字符串
本文介绍了JavaScript中的字符串处理,包括普通字符串和模板字符串的使用方法及常见字符串操作方法如`charAt`、`concat`、`endsWith`等,适合前端学习者参考。作者是一位热爱前端技术的大一学生,专注于分享实用的编程技巧。
35 2
|
2月前
|
存储 JavaScript 前端开发
JavaScript学习第一章
本文档介绍了JavaScript的基础知识,包括其在网页中的作用、如何通过JavaScript动态设置HTML元素的CSS属性,以及JavaScript中的变量类型(`var`、`let`、`const`)和数据类型(基本数据类型与引用数据类型)。通过实例代码详细解释了JavaScript的核心概念,适合初学者入门学习。
57 1
|
2月前
|
前端开发 JavaScript 数据处理
CSS 变量的作用域和 JavaScript 变量的作用域有什么不同?
【10月更文挑战第28天】CSS变量和JavaScript变量虽然都有各自的作用域概念,但由于它们所属的语言和应用场景不同,其作用域的定义、范围、覆盖规则以及与其他语言特性的交互方式等方面都存在明显的差异。理解这些差异有助于更好地在Web开发中分别运用它们来实现预期的页面效果和功能逻辑。
|
2月前
|
JavaScript 前端开发
如何在 JavaScript 中实现块级作用域?
【10月更文挑战第29天】通过使用 `let`、`const` 关键字、立即执行函数表达式以及模块模式等方法,可以在JavaScript中有效地实现块级作用域,更好地控制变量的生命周期和访问权限,提高代码的可维护性和可读性。
|
2月前
|
JavaScript 前端开发
javascript的作用域
【10月更文挑战第19天javascript的作用域
|
3月前
|
JavaScript 前端开发
JavaScript 作用域
JavaScript 作用域是指程序中可访问的变量、对象和函数的集合。它分为函数作用域和局部作用域。函数作用域内的变量仅在函数内部可见,而全局作用域的变量在整个网页中均可访问。局部变量在函数执行完毕后会被销毁,而全局变量则在整个脚本生命周期中都存在。未使用 `var` 关键字声明的变量默认为全局变量。
|
3月前
|
JavaScript
js学习--商品列表商品详情
js学习--商品列表商品详情
34 2