103 Stimmen

Javascript-Funktion scoping und hoisting

Ich habe gerade einen tollen Artikel gelesen über JavaScript-Scoping und Heben von Ben Cherry in dem er das folgende Beispiel anführt:

var a = 1;

function b() {
    a = 10;
    return;

    function a() {}
}
b();
alert(a);

Bei Verwendung des obigen Codes meldet der Browser "1".

Ich bin mir immer noch nicht sicher, warum er eine "1" zurückgibt. Einige der Dinge, die er sagt, kommen mir in den Sinn wie: Alle Funktionsdeklarationen werden nach oben gehievt. Man kann eine Variable mit einer Funktion in den Gültigkeitsbereich bringen. Trotzdem verstehe ich es nicht.

0voto

Nam V. Do Punkte 551

Soweit ich weiß, geschieht das Hailing zum Beispiel bei der Deklaration von Variablen und Funktionen:

a = 7;
var a;
console.log(a) 

Was im Inneren der JavaScript-Engine passiert:

var a;
a = 7;
console.log(a);
// 7

Oder:

console.log(square(7)); // Output: 49
function square(n) { return n * n; }

Sie wird es werden:

function square(n) { return n * n; }
console.log(square(7)); // 49

Aber Zuweisungen wie die Zuweisung von Variablen und Funktionsausdrücken werden nicht angehoben: Zum Beispiel:

console.log(x);
var x = 7; // undefined

Es könnte so aussehen:

var x;
console.log(x); // undefined
x = 7;

0voto

Lord Punkte 5685

Um das Hosting in Javascript in einem Satz zu beschreiben, werden Variablen und Funktionen an die Spitze des Bereichs gehievt, in dem sie deklariert sind.

enter image description here

Ich gehe davon aus, dass Sie ein Anfänger sind. Um das Heben richtig zu verstehen, müssen wir zuerst den Unterschied verstehen zwischen undefiniert y ReferenceError

 var v;
 console.log(v);
 console.log(abc);
/*
The output of the above codes are:
undefined
ReferenceError: abc is not defined*/

Im folgenden Code sehen wir, dass eine Variable und ein Funktionsausdruck deklariert sind.

<script>
var totalAmo = 8;
var getSum = function(a, b){
      return a+b;
}
</script>

aber das wirkliche Bild mit dem Beweis, dass sowohl die Variable als auch die Funktion auf den oberen Teil des Anwendungsbereichs gehievt werden:

console.log(totalAmo);
console.log(getSum(8,9));
var totalAmo = 8;
var getSum = function(a, b){
      return a+b;
}
console.log(totalAmo);
console.log(getSum(9,7));

Die Ausgabe der ersten beiden Protokolle lautet undefiniert y TypeError: getSum ist keine Funktion weil sowohl var totalAmo y getSum werden wie ein Blasebalg auf den oberen Rand ihres Geltungsbereichs gehievt

 <script>
        var totalAmo;
        var getSum;

        console.log(totalAmo);
        console.log(getSum(8,9));
        var totalAmo = 8;
        var getSum = function(a, b){
            return a+b;
        }
        console.log(totalAmo);
        console.log(getSum(9,7));
    </script>

Aber für Funktionen Erklärung ganze Funktionen auf der Oberseite ihres Anwendungsbereichs gehängt.

console.log(getId());
function getId(){
   return 739373;
}
/* output: 739373, because the whole function hoisted on the top of the scope.*/

Die gleiche Logik gilt nun für die innerhalb des Funktionsbereichs deklarierten Variablen, Funktionsausdrücke und Funktionsdeklarationen. Wichtigster Punkt: Sie werden nicht an die Spitze der Akte gehievt ;

function functionScope(){
            var totalAmo;
            var getSum;

            console.log(totalAmo);
            console.log(getSum(8,9));
            var totalAmo = 8;
            var getSum = function(a, b){
                return a+b;
            }
        }

Wenn Sie also var Schlüsselwort, Variable und Funktion, die in den oberen Bereich des Bereichs (globaler Bereich und Funktionsbereich) gehievt werden. Was ist mit lassen Sie y const const- und let-Variablen kennen immer noch den globalen Bereich und den Funktionsbereich, genau wie var, aber const- und let-Variablen kennen auch einen anderen Bereich, den blockierten Bereich. Ein Blockbereich ist immer dann vorhanden, wenn es einen Code-Block gibt, z. B. eine for-Schleife, eine if else-Anweisung, eine while-Schleife usw.

Wenn wir const und let verwenden, um eine Variable in diesem Blockbereich zu deklarieren, wird die Variablendeklaration nur an der Spitze des Blocks, in dem sie sich befindet, gehostet, und nicht an der Spitze der übergeordneten Funktion oder an der Spitze des globalen Bereichs, in dem sie gehostet wird.

 function getTotal(){
            let total=0;
            for(var i = 0; i<10; i++){
                let valueToAdd = i;
                var multiplier = 2;
                total += valueToAdd*multiplier;
            }
            return total;
        }

Die Variablen im Adobe-Beispiel werden wie folgt hochgezogen

 function getTotal(){
            let total;
            var multiplier;
            total = 0;
            for(var i = 0; i<10; i++){
                let valueToAdd;
                valueToAdd = i;
                multiplier = 2;
                total += valueToAdd*multiplier;
            }
            return total;
        }

0voto

xgqfrms Punkte 7010

ES5: Funktionsaufzug und variabler Aufzug

function hoisting Priorität ist greater als variable hoisting

"use strict";

/**
 *
 * @author xgqfrms
 * @license MIT
 * @copyright xgqfrms
 * @created 2016-06-01
 * @modified
 *
 * @description function-hoisting.js
 * @augments
 * @example
 * @link
 *
 */

(function() {
  const log = console.log;

  var a = 1;
  function b() {
    a = 10;
    log(`local a`, a)
    return;
    // function hoisting priority is greater than variable hoisting
    function a() {}
  }
  b();
  log(`global a`, a);
  // local a 10
  // global a 1
})();

was gleich ist mit

(function() {
  const log = console.log;

  // define "a" in global scope
  var a = 1;
  function b() {
    // define "a" in local scope
    var a ;
    // assign function to a
    a = function () {};
    // overwrites local variable "a"
    a = 10;
    log(`local a`, a);
    return;
  }

  b();
  // log global variable "a"
  log(`global a`, a);

  // local a 10
  // global a 1
})();

der Grund für das Heben

var a = 1;                
//"a" is global scope
function b() {  
   var a = function () {}; 
   //"a" is local scope 
   var x = 12; 
   //"x" is local scope 
   a = 10;
   //global variable "a" was overwrited by the local variable "a"  
   console.log("local a =" + a);
   return console.log("local x = " + x);
}       
b();
// local a =10
// local x = 12
console.log("global a = " + a);
// global a = 1
console.log("can't access local x = \n");
// can't access local x = 
console.log(x);
// ReferenceError: x is not defined

/**
 *  scpope & closure & hoisting (var/function)
 *  
 * 1. scpope : the global var can be access in any place(the whole file scope), local var only can be accessed by the local scope(function/block scope)!
 * Note: if a local variable not using var keywords in a function, it will become a global variable!
 * 
 * 2. closure : a function inner the other function, which can access local scope(parent function) & global scope, howerver it's vars can't be accessed by others! unless, your return it as return value!
 * 
 * 3. hoisting : move all declare/undeclare vars/function to the scope top, than assign the value or null!
 * Note: it just move the declare, not move the value!
 * 
 */

ES6 let , const kein Heben vorhanden

(() => {
  const log = console.log;
  log(a)
  // Error: Uncaught ReferenceError: Cannot access 'a' before initialization
  let a = 1;
})();

(() => {
  const log = console.log;
  log(b)
  // Error: Uncaught ReferenceError: Cannot access 'b' before initialization
  const b = 1;
})();

refs

https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/var

https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/let

https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/const

CodeJaeger.com

CodeJaeger ist eine Gemeinschaft für Programmierer, die täglich Hilfe erhalten..
Wir haben viele Inhalte, und Sie können auch Ihre eigenen Fragen stellen oder die Fragen anderer Leute lösen.

Powered by:

X