Maison >interface Web >js tutoriel >Explication détaillée de la portée et de la chaîne de portée en JavaScript

Explication détaillée de la portée et de la chaîne de portée en JavaScript

青灯夜游
青灯夜游avant
2020-10-27 18:02:052275parcourir

Explication détaillée de la portée et de la chaîne de portée en JavaScript

JavaScript possède une fonctionnalité appelée scope. Bien que pour de nombreux développeurs novices, le concept de portée ne soit pas facile à comprendre. Dans cet article, je ferai de mon mieux pour expliquer la portée et la chaîne de portée de la manière la plus simple. J'espère que tout le monde apprendra quelque chose !

Portée

1. Qu'est-ce que la portée

La portée est une variable dans une partie spécifique du code d'exécution, l'accessibilité des fonctions et des objets. En d’autres termes, la portée détermine la visibilité des variables et autres ressources au sein d’un bloc de code. Peut-être que ces deux phrases ne sont pas faciles à comprendre. Jetons d'abord un exemple :

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

À partir de l'exemple ci-dessus, vous pouvez comprendre le concept de portée La variable inVariable n'est pas déclarée dans la portée globale, il est donc dans la portée globale. Une erreur sera signalée si la valeur est récupérée. On peut le comprendre ainsi : Le scope est un territoire indépendant qui empêche les variables de fuir ou d'être exposées. En d'autres termes, la plus grande utilisation de la portée est d'isoler les variables portant le même nom dans différentes portées ne seront pas en conflit.

Avant ES6, JavaScript n'avait pas de portée au niveau du bloc, seulement une portée globale et une portée de fonction. L'arrivée d'ES6 nous offre une « portée au niveau du bloc », qui peut être reflétée par les nouvelles commandes let et const.

2. Portée globale et portée de la fonction

Les objets accessibles n'importe où dans le code ont une portée globale. De manière générale, les situations suivantes ont une portée globale :

    La fonction la plus externe et les variables définies en dehors de la fonction la plus externe ont une portée globale
  • 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
    Toutes les dernières définitions sont affectées directement Les variables sont automatiquement déclarées avoir une portée globale
  • function outFun2() {
        variable = "未定义直接赋值的变量";
        var inVariable2 = "内层变量2";
    }
    outFun2();//要先执行这个函数,否则根本不知道里面是啥
    console.log(variable); //未定义直接赋值的变量
    console.log(inVariable2); //inVariable2 is not defined
    Toutes les propriétés de l'objet window ont une portée globale
Généralement, les propriétés intégrées de l'objet window ont une portée globale, comme window.name , fenêtre.emplacement, fenêtre.top, etc.

La portée globale a un inconvénient : si nous écrivons de nombreuses lignes de code JS et que les définitions de variables ne sont pas incluses dans les fonctions, alors elles seront toutes dans la portée globale. Cela polluera l'espace de noms global et provoquera facilement des conflits de noms.

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

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

C'est pourquoi le code source des bibliothèques telles que jQuery et Zepto sera placé dans

. Parce que toutes les variables placées à l'intérieur ne seront pas divulguées ou exposées, ne seront pas polluées vers l'extérieur et n'affecteront pas les autres bibliothèques ou scripts JS. Il s’agit d’une manifestation de l’étendue des fonctions. (function(){....})()

La portée de la fonction fait référence aux variables déclarées à l'intérieur d'une fonction. Contrairement à la portée globale, la portée locale n'est généralement accessible que dans un fragment de code fixe, le plus souvent dans une fonction.

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

Les portées sont hiérarchiques. La portée interne peut accéder aux variables de la portée externe, mais pas l'inverse. Regardons un exemple. Il peut être plus facile de comprendre en utilisant des bulles comme métaphore de la portée :

Explication détaillée de la portée et de la chaîne de portée en JavaScript

Le résultat final est 2, 4, 12

Bulle 1 est la portée globale, avec l'identifiant foo
  • Bulle 2 est la portée foo, avec les identifiants a, bar, b
  • Bulle 3 est la barre de portée ; , seul identifiant c.
  • Il convient de noter :
Les instructions de bloc (instructions entre accolades "{}"), telles que les instructions conditionnelles if et switch ou les instructions de boucle for et while, contrairement aux fonctions, elles ne le font pas Un nouveau scope

sera créé. Les variables définies dans une instruction de bloc resteront dans la portée dans laquelle elles existent déjà.

if (true) {
    // 'if' 条件语句块不会创建一个新的作用域
    var name = 'Hammad'; // name 依然在全局作用域中
}
console.log(name); // logs 'Hammad'
Les débutants en JS ont souvent besoin d'un certain temps pour s'habituer au levage variable, et s'ils ne comprennent pas ce comportement unique, cela peut conduire à des

bugs. Pour cette raison, ES6 a introduit une portée au niveau du bloc pour rendre le cycle de vie des variables plus contrôlable.


3. La portée au niveau du bloc

La portée au niveau du bloc peut être déclarée via les nouvelles commandes let et const. Les variables déclarées ne peuvent pas être déclarées en dehors de la portée du. accès spécifié. Des portées au niveau du bloc sont créées :

À l'intérieur d'une fonction
  1. À l'intérieur d'un bloc de code (entouré par une paire d'accolades)
  2. let est cohérente avec celle de var. Vous pouvez essentiellement utiliser let au lieu de var pour la déclaration de variable, mais cela limitera la portée de la variable au bloc de code actuel. La portée au niveau du bloc présente les caractéristiques suivantes :

Les variables déclarées ne seront pas promues en haut du bloc de code

  • La déclaration let/const ne sera pas promue en haut du bloc de code. bloc de code actuel en haut, vous devez donc placer manuellement la déclaration let/const en haut pour rendre la variable disponible dans tout le bloc de code.
  • function getValue(condition) {
    if (condition) {
    let value = "blue";
    return value;
    } else {
    // value 在此处不可用
    return null;
    }
    // value 在此处不可用
    }

Les déclarations en double sont interdites

  • Si un identifiant a été défini à l'intérieur d'un bloc de code, alors l'utilisation du même identifiant pour une déclaration let dans ce bloc de code provoquera un erreur à lancer. Par exemple :
  • 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视频教程

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

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer