ES6
Block-level scoped binding
var
And variable promotion
Previously, JS used var to declare variables. Whether in function scope or global scope,var declaration has variable promotion mechanism, that is, will be declared at the top of the current scope variable. Code:
function getValue(condition){ if(condition){ var value = "blue"; return value; } else { return null; }}Copy the code
In fact, this code in the JS engine precompile phase is equivalent to:
function getValue(condition){ var value; // The variable is promoted to the top of the function scope if(condition){value = "blue"; return value; } else { return null; }}Copy the code
That’s called variable promotion.
In addition to variable promotion, there is also function promotion. There are three ways to create functions in javascript :Function constructors, Function declarations, and Function expressions. Function promotion exists only for functions created by Function declarations.
Function declaration:
function fun(x){ alert(x); } Copy the code
Function expression:
Var fun = function(x){alert(x); Var fun = function fc(x){alert(x); }Copy the code
Function constructor:
var multiply = new Function('x', 'y', 'return x * y'); Copy the code
We can see that, except for the function created by the function declaration, all the other methods assign the created function to the variable, so the other methods actually do variable promotion.
Example of function promotion:
Console. log(a); a(); function a() { console.log(3); } function a() {console.log(3); } console.log(a); a(); // Print function first and then print 3Copy the code
When variable promotion and function promotion exist together, which has higher priority? Look at this example:
console.log(multiply); console.log(multiply()); multiply = 2; function multiply() { console.log(10); }; console.log(multiply); console.log(multiply()); Copy the code
The method and variable have the same name. Let’s run it to see what it prints:
ƒ multiply () {the console. The log (10); } 10 2 Uncaught TypeError: multiply is not a function at <anonymous>:8:13Copy the code
Variable promotion can cause problems
1. Variables are easy to overwrite.
var tmp=new Date(); function f(){ console.log(tmp); if(false){ var tmp='hello world'; } } f(); // The function returns undefinedCopy the code
The function is expected to print a time, but in fact, due to the variable promotion, the code actually becomes:
var tmp=new Date();
function f(){
var tmp;
console.log(tmp);
if(false){
tmp='hello world';
}
}
f();
Copy the code
A variable promotion causes the TMP in the function to override the TMP in the global scope. This is just an example to better understand “a variable promotion causes a variable to be overwritten “. In the actual writing of the code, we should not declare twice, nor should we declare two variables with different meanings with the same name. That, in itself, violates the coding specification.
2. Variable not destroyed This is a very common code:
function foo(){
for (var i = 0; i < 7; i++) {
}
console.log(i);
};
foo();
Copy the code
In general, it’s easy to assume that the I in the for loop is destroyed after execution, but when we execute this code, we see that I is printed out as 7, which is not what we expected because I is promoted to the top of the function scope during the precompilation phase and remains there for the duration of the function execution.
Incidentally, we can also understand that in previous JS we said that the scope was only function scope and global scope, because the variable promotion mechanism makes variables inside a function actually declared at the top of the function. In ES6, const and let declarations were introduced, and there was no variable promotion, hence the concept of block-level scope.
const
let
And block-level scope
To eliminate the negative impact of var declarations on programs,ES6 introduces block-level scopes to control the life cycle of variables.
Block-level scopes exist in
- Function of the internal
{
and}
between
let
The let declaration is not promoted, so we generally declare it at the top of the block-level scope and limit the scope of the variables to the current code block (between {and}). Let prohibits repeated declarations and does not promote variables, which is a double insurance compared to var. See the examples:
var count = 30; let count = 10; Uncaught SyntaxError: Identifier 'count' has already been declaredCopy the code
At the same time:
var count = 30;
if(condition){
let count = 50;
console.log(count);
}
Copy the code
Count declared by let in the block-level scope of an if statement does not throw an exception because count is declared by var in the outer scope, and the outer scope is masked.
const
Const is otherwise identical to let, but with two notable differences:
- It is used to declare constants, so it must be initialized.
const name = 'xlx'; const age; // Uncaught SyntaxError: Missing initializer in const declarationCopy the code
- No matter inStrict ModeorSloppy ModeDo not take it for granted
const
The declared constant is reassigned.
const age = 1; age = 2; TypeError: Assignment to constant variable at <anonymous>:2:5Copy the code
Strict mode: Makes Javascript run under stricter conditions
- Strict mode eliminates some of the original silent errors by throwing errors
- Strict mode fixes some of the flaws that make it difficult for JavaScript engines to perform optimizations: sometimes, the same code can run faster in strict mode than in non-strict mode
- Strict mode disables some syntax that may be defined in future versions of ECMAScript
While a const declaration cannot be modified, it can be modified if the variable is an object, because the ++const declaration does not allow binding changes, but it can change the value ++.
const person = { name:"xlx" }; person.name = "xax"; Person = {name:"xax"}; TypeError: Assignment toconstant variable at <anonymous>:5:8Copy the code
TDZ (Temporal Dead Zone)
Let’s start with some code:
if(condition){
console.log(value);
var value = "blue";
}
Copy the code
According to the variable promotion mechanism, we know that this code prints undefined. Let’s replace var.
if(condition){
console.log(value);
let value = "blue";
}
Copy the code
It prints an error line:
Uncaught ReferenceError: Cannot access 'value' before initialization at <anonymous>:2:17
Copy the code
Same thing with const. Let and const are not declared in scope because they do not have a variable promotion mechanism. According to the ES6 standard:
Let and const declarations define variables that are scoped to the running execution context’s LexicalEnvironment. The variables are created when their containing Lexical Environment is instantiated but may not be accessed in any way until The variable’s LexicalBinding is evaluated. A variable defined by A LexicalBinding with an Initializer is assigned the Value of its Initializer’s AssignmentExpression when the LexicalBinding is Evaluated, not when the variable is created. If a LexicalBinding in a let declaration does not have an Initializer the variable is assigned the value undefined when the LexicalBinding is evaluated.
Let and const create a block scope whose variables are created at the Lexical Environment instantiation stage, but not yet LexicalBinding. During this time, the variable cannot be accessed. Access means throwing an exception. We call it a temporary dead zone.
In the example code above,let value = “blue” in the block scope wrapped around {}; The previous space was a temporary dead zone. The js engine was scanning console.log(value); Although the value variable has been created, it is in a temporary dead zone because no variable declaration statement has been executed. If a variable is accessed before the variable declaration statement is executed, an error is reported.
Block-level scoped binding in loops
As mentioned earlier, the VAR in the for loop prevents I from being destroyed at the end of the loop and is accessed outside the loop, which is not what we expect. In addition, there is an example that illustrates this problem. In this example, the goal is to access every I, but because of the variable increment of var, there is a problem. Code:
var func = []; for(var i; i<10; i++){ func.push(function(){ console.log(i); }); }; func.forEach(function(fc){ fc(); });Copy the code
We expected to print 0-9, of course, but printed all the numbers as 10. The reason for this is that the var promotion declares I at the top of the global scope, and every time the function I is created, it refers to the same reference. The value of I is changed, and the I in all the functions is changed to 10.
I’ll just replace it with let.
In factlet
infor
The behavior in the loop is inES6
What is specifically defined in the standard is not just what we think of as “no variable promotion”. Each time through the loop, a new one is createdlet
variablei
, and initialize to currenti
Phi, so there are three completely independent phii
, created inside the loopfunction
And you can get your owni
The value of the.
Const behaves like let in a loop, except that const declares a constant, so in code like this:
for(const i; i<10; i++){ func.push(function(){ console.log(i); }); };Copy the code
Const I = const I; const I = const I; const I = const I; const I = const I; However, in code like this:
for(const i in object){
func.push(function(){
console.log(i);
});
};
Copy the code
, because it creates three separate I’s at a time and does not change the value of I each time through the loop.
Global block-level scoped binding
There is another big difference in the behavior of const var lets in the global scope. Variables declared by var in the global scope become properties of the global object. This means that an existing property of the Window object is overridden by var. If you use let and const, a new binding is created in the global scope, but the binding is not added to the window and is not overwritten. Code examples:
let RegExp = "hello"; console.log(RegExp); //hello console.log(window.RegExp === RegExp); //false console.log("RegExp" in window); //true let xlx = "xlx"; console.log("xlx" in window); //falseCopy the code
Best practices
Use the default
const
, used only when you really need to change a variablelet
Most variable values should not change after initialization, and unexpected variable value changes are the source of many bugs.