>  기사  >  웹 프론트엔드  >  JavaScript의 범위 및 범위 체인에 대한 자세한 설명

JavaScript의 범위 및 범위 체인에 대한 자세한 설명

青灯夜游
青灯夜游앞으로
2020-10-27 18:02:052195검색

JavaScript의 범위 및 범위 체인에 대한 자세한 설명

JavaScript에는 범위라는 기능이 있습니다. 많은 초보 개발자에게는 범위의 개념이 이해하기 쉽지 않지만, 이 기사에서는 범위와 범위 체인을 가장 간단한 방법으로 설명하려고 노력할 것입니다.

Scope

1. 범위란 무엇인가요?

범위는 런타임 코드의 특정 부분에 있는 변수, 함수 및 개체에 대한 접근성입니다. 즉, 범위는 코드 블록 내의 변수 및 기타 리소스의 가시성을 결정합니다. 아마도 이 두 문장은 이해하기 쉽지 않을 것입니다. 먼저 예를 살펴보겠습니다.

function outFun2() {
    var inVariable = "内层变量2";
}
outFun2();//要先执行这个函数,否则根本不知道里面是啥
console.log(inVariable); // Uncaught ReferenceError: inVariable is not defined

위의 예를 보면 inVariable 변수가 전역 범위에서 선언되지 않았으므로 범위의 개념을 이해할 수 있습니다. 전역 범위는 오류 보고입니다. 범위는 독립적인 영역이므로 변수가 유출되거나 노출될 수 없습니다. 즉, 범위의 가장 큰 용도는 서로 다른 범위에서 동일한 이름을 가진 변수를 격리하는 것입니다.

ES6 이전에는 JavaScript에 블록 수준 범위가 없었고 전역 범위와 함수 범위만 있었습니다. ES6의 등장으로 '블록 수준 범위'가 제공되었으며, 이는 새로운 명령 let 및 const에 반영될 수 있습니다.

2. 전역 범위 및 함수 범위

코드 내 어디에서나 액세스할 수 있는 개체에는 전역 범위가 있습니다. 일반적으로 다음과 같은 상황에 전역 범위가 있습니다.

    가장 바깥쪽 함수 및 가장 바깥쪽 함수 외부에 정의된 변수 전역 범위
  • var outVariable = "我是最外层变量"; //最外层变量
    function outFun() { //最外层函数
        var inVariable = "内层变量";
        function innerFun() { //内层函数
            console.log(inVariable);
        }
        innerFun();
    }
    console.log(outVariable); //我是最外层变量
    outFun(); //内层变量
    console.log(inVariable); //inVariable is not defined
    innerFun(); //innerFun is not defined
    정의되지 않고 직접 할당된 모든 변수는 자동으로 전역 범위를 갖도록 선언됩니다
  • function outFun2() {
        variable = "未定义直接赋值的变量";
        var inVariable2 = "内层变量2";
    }
    outFun2();//要先执行这个函数,否则根本不知道里面是啥
    console.log(variable); //未定义直接赋值的变量
    console.log(inVariable2); //inVariable2 is not defined
    창 개체의 모든 속성에는 전역 범위가 있습니다
일반적으로 창 개체의 내장 속성에는 전역 범위가 있습니다. window.name, window.location, window.top 등과 같은 범위.

전역 범위에는 단점이 있습니다. JS 코드를 여러 줄로 작성하고 변수 정의가 함수에 포함되지 않으면 모두 전역 범위에 속하게 됩니다. 이로 인해 전역 네임스페이스가 오염되고 이름 충돌이 쉽게 발생합니다.

// 张三写的代码中
var data = {a: 100}

// 李四写的代码中
var data = {x: true}

이것이 jQuery, Zepto 및 기타 라이브러리의 소스 코드가

에 배치되는 이유입니다. 내부에 배치된 모든 변수는 유출되거나 노출되지 않으므로 외부로 오염되지 않으며, 다른 라이브러리나 JS 스크립트에 영향을 미치지 않습니다. 이는 함수 범위의 표현입니다. (function(){....})()

함수 범위는 함수 내부에 선언된 변수를 참조합니다. 전역 범위와 달리 로컬 범위는 일반적으로 고정된 코드 조각 내에서만 액세스할 수 있으며 가장 일반적으로 함수 내에서 액세스할 수 있습니다.

function doSomething(){
    var blogName="浪里行舟";
    function innerSay(){
        alert(blogName);
    }
    innerSay();
}
alert(blogName); //脚本错误
innerSay(); //脚本错误

범위는 계층적입니다. 내부 범위는 외부 범위의 변수에 액세스할 수 있지만 그 반대는 불가능합니다. 예를 살펴보겠습니다. 버블을 범위에 대한 비유로 사용하는 것이 더 쉬울 수 있습니다.

JavaScript의 범위 및 범위 체인에 대한 자세한 설명

최종 출력 결과는 2, 4, 12

    Bubble 1은 식별자 foo가 있는 전역 범위입니다.
  • Bubble 2는 식별자 a, bar 및 b가 있는 범위 foo입니다.
  • Bubble 3은 식별자 c만 있는 범위 bar입니다.
다음에 주목할 가치가 있습니다.

if 및 스위치 조건문 또는 for 및 while 루프 문과 같은 블록 문(중괄호 "{}" 사이의 문)은 함수와 달리 새로운 범위를 생성하지 않습니다 . 블록 문 내에 정의된 변수는 이미 존재하는 범위에 유지됩니다.

if (true) {
    // 'if' 条件语句块不会创建一个新的作用域
    var name = 'Hammad'; // name 依然在全局作用域中
}
console.log(name); // logs 'Hammad'

JS 초보자는 변수 호이스팅에 익숙해지는 데 시간이 필요한 경우가 많으며, 이 고유한 동작을 이해하지 못하면

버그가 발생할 수 있습니다. 이 때문에 ES6에서는 변수의 수명주기를 보다 쉽게 ​​제어할 수 있도록 블록 수준 범위를 도입했습니다.

3. 블록 수준 범위

새로운 명령 let 및 const를 통해 선언된 변수는 지정된 블록 범위 외부에 액세스할 수 없습니다. 블록 수준 범위가 생성됩니다.

    함수 내부
  1. 코드 블록 내부(중괄호 쌍으로 둘러싸여 있음)
let 선언의 구문은 var의 구문과 일치합니다. 기본적으로 변수 선언에 var 대신 let을 사용할 수 있지만 변수의 범위는 현재 코드 블록으로 제한됩니다. 블록 수준 범위에는 다음과 같은 특징이 있습니다.

    변수 선언은 코드 블록의 최상위로 승격되지 않습니다.
let/const 선언은 현재 코드 블록의 최상위로 승격되지 않으므로 수동으로 선언해야 합니다. let/const를 선언하십시오. 코드 블록 전체에서 변수를 사용할 수 있도록 하려면 이를 맨 위에 배치하십시오.

function getValue(condition) {
if (condition) {
let value = "blue";
return value;
} else {
// value 在此处不可用
return null;
}
// value 在此处不可用
}

    중복 선언은 금지됩니다
코드 블록 내에 식별자가 이미 정의된 경우 이 코드 블록 내의 let 선언에 동일한 식별자를 사용하면 오류가 발생합니다. 예:

var count = 30;
let count = 40; // Uncaught SyntaxError: Identifier 'count' has already been declared

在本例中, count 变量被声明了两次:一次使用 var ,另一次使用 let 。因为 let 不能在同一作用域内重复声明一个已有标识符,此处的 let 声明就会抛出错误。但如果在嵌套的作用域内使用 let 声明一个同名的新变量,则不会抛出错误。

var count = 30;
// 不会抛出错误
if (condition) {
let count = 40;
// 其他代码
}
  • 循环中的绑定块作用域的妙用

开发者可能最希望实现 for 循环的块级作用域了,因为可以把声明的计数器变量限制在循环内,例如,以下代码在 JS 经常见到:

<button>测试1</button>
<button>测试2</button>
<button>测试3</button>
<script type="text/javascript">
   var btns = document.getElementsByTagName(&#39;button&#39;)
    for (var i = 0; i < btns.length; i++) {
      btns[i].onclick = function () {
        console.log(&#39;第&#39; + (i + 1) + &#39;个&#39;)
      }
    }
</script>

我们要实现这样的一个需求: 点击某个按钮, 提示"点击的是第 n 个按钮",此处我们先不考虑事件代理,万万没想到,点击任意一个按钮,后台都是弹出“第四个”,这是因为 i 是全局变量,执行到点击事件时,此时 i 的值为 3。那该如何修改,最简单的是用 let 声明 i

 for (let i = 0; i < btns.length; i++) {
    btns[i].onclick = function () {
      console.log(&#39;第&#39; + (i + 1) + &#39;个&#39;)
    }
  }

作用域链

1.什么是自由变量

首先认识一下什么叫做 自由变量 。如下代码中,console.log(a)要得到 a 变量,但是在当前的作用域中没有定义 a(可对比一下 b)。当前作用域没有定义的变量,这成为 自由变量 。自由变量的值如何得到 —— 向父级作用域寻找(注意:这种说法并不严谨,下文会重点解释)。

var a = 100
function fn() {
    var b = 200
    console.log(a) // 这里的a在这里就是一个自由变量
    console.log(b)
}
fn()

2. 什么是作用域链

如果父级也没呢?再一层一层向上寻找,直到找到全局作用域还是没找到,就宣布放弃。这种一层一层的关系,就是 作用域链 。

var a = 100
function F1() {
    var b = 200
    function F2() {
        var c = 300
        console.log(a) // 自由变量,顺作用域链向父作用域找
        console.log(b) // 自由变量,顺作用域链向父作用域找
        console.log(c) // 本作用域的变量
    }
    F2()
}
F1()

3. 关于自由变量的取值

关于自由变量的值,上文提到要到父作用域中取,其实有时候这种解释会产生歧义。

var x = 10
function fn() {
  console.log(x)
}
function show(f) {
  var x = 20
  (function() {
    f() //10,而不是20
  })()
}
show(fn)

在 fn 函数中,取自由变量 x 的值时,要到哪个作用域中取?——要到创建 fn 函数的那个作用域中取,无论 fn 函数将在哪里调用

所以,不要在用以上说法了。相比而言,用这句话描述会更加贴切:要到创建这个函数的那个域”。
作用域中取值,这里强调的是“创建”,而不是“调用”
,切记切记——其实这就是所谓的"静态作用域"

var a = 10
function fn() {
  var b = 20
  function bar() {
    console.log(a + b) //30
  }
  return bar
}
var x = fn(),
  b = 200
x() //bar()

fn()返回的是 bar 函数,赋值给 x。执行 x(),即执行 bar 函数代码。取 b 的值时,直接在 fn 作用域取出。取 a 的值时,试图在 fn 作用域取,但是取不到,只能转向创建 fn 的那个作用域中去查找,结果找到了,所以最后的结果是 30

作用域与执行上下文

许多开发人员经常混淆作用域和执行上下文的概念,误认为它们是相同的概念,但事实并非如此。

我们知道 JavaScript 属于解释型语言,JavaScript 的执行分为:解释和执行两个阶段,这两个阶段所做的事并不一样:

解释阶段:

  • 词法分析
  • 语法分析
  • 作用域规则确定

执行阶段:

  • 创建执行上下文
  • 执行函数代码
  • 垃圾回收

JavaScript 解释阶段便会确定作用域规则,因此作用域在函数定义时就已经确定了,而不是在函数调用时确定,但是执行上下文是函数执行之前创建的。执行上下文最明显的就是 this 的指向是执行时确定的。而作用域访问的变量是编写代码的结构确定的。

作用域和执行上下文之间最大的区别是:
执行上下文在运行时确定,随时可能改变;作用域在定义时就确定,并且不会改变

一个作用域下可能包含若干个上下文环境。有可能从来没有过上下文环境(函数从来就没有被调用过);有可能有过,现在函数被调用完毕后,上下文环境被销毁了;有可能同时存在一个或多个(闭包)。同一个作用域下,不同的调用会产生不同的执行上下文环境,继而产生不同的变量的值

相关免费学习推荐:js视频教程

更多编程相关知识,请访问:编程入门!!

위 내용은 JavaScript의 범위 및 범위 체인에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 cnblogs.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제