1. What is scope
1.1 Concepts
Syntax analysis and code generation
Responsible for compiling and executing JS program
Collect and maintain queries consisting of declared variables that determine access to the currently executing code
Who is the target of the assignment operation (to whom to assign)
Who is the source of the assignment operation (to get the value of a variable)
var a = 2
Copy the code
Let’s see how these concepts fit together. Right
(1) The compiler first declares a variable in the current scope (if it has not been declared before);
- Word segmentation/lexical analysis: The compiler first breaks a string of characters into meaningful (for the language) fragments called tokens, such as var a = 2; . Var, a, =, 2.
- Parsing/parsing: The compiler converts a stream (array) of tokens into an AST — Abstract Syntax Tree, which represents the Syntax structure of the program.
- Code generation: The compiler translates the abstract syntax tree generated in the previous step into machine instructions for the engine to execute.
(2) Then at runtime the engine looks for the variable == in scope == and assigns it == if it can find it;
- LHS (left-hand Side) and RHS (Right-hand Side) are two ways for JS engine to operate variables in the code execution stage. The difference between them is whether the query purpose of variables is variable assignment or query.
- LHS can be understood as a variable to the left of the assignment operator (=), such as a = 2. The current engine search for the variable a == is to assign the variable ==. In this case, the engine doesn’t care what the original value of variable A is, it just assigns the value 2 to variable A.
- RHS can be understood as a variable to the right of the assignment operator (=), such as console.log(a), where the engine == searches for variable A to find out what the actual value of variable A is before printing it out.
It takes these two steps to complete the assignment of a variable.
1.2 What is scope
One of the most basic models of almost any language is to store values in variables and later retrieve or modify them. The ability to store and retrieve values in variables gives a program state. This raises two questions: where are these variables stored? How does the program find them when it needs them? Answering these questions requires a well-defined set of rules that define how variables are stored and how to find them. We call this set of rules: scope.
Brevity: == refers to the area of the program where variables are defined, which determines how much access the currently executing code has to the variables. = =
For the most part in javascript, there are only two scope types (ES5 standard) :
- Global scope: A global scope is the outermost scope of a program and always exists.
- Function scope: A function scope is created only when the function is defined, within the parent function scope/global scope.
/* global scope starts */ var a = 1; Function func () {/* var a = 2; console.log(a); } /* func function scope end */ func(); // => 2 console.log(a); // => 1 /* Global scope ends */Copy the code
Due to scope limitations, each independently executing code block can only access variables in its own scope and in the outer scope, but not in the inner scope.
When function nesting occurs, scope chains are created;
1.3 Scope chain
function foo(a) { var b = a * 2; function bar(c) { console.log( a, b, c ); } bar(b * 3); } foo(2); / / 2, 4 12Copy the code
Based on the previous knowledge, we know that inside the bar function, three RHS queries will be made to obtain the values of three variables a, B and C respectively. The internal scope of bar can only fetch the value of variable C; both a and b are fetched from the scope of the external function foo.
When the executable code accesses a variable internally, it looks in the local scope first, returns if it finds the target variable, and continues in the parent scope otherwise… Go all the way to the global scope. We refer to this nesting mechanism of scopes as scope chains.
1.4.1 Lexical scope
Lexical Scopes are the type of scope used in javascript, and Lexical Scopes can also be called static Scopes, as opposed to dynamic Scopes. So what’s the difference between lexical and dynamic scoping that javascript uses? Take a look at this code:
var value = 1; function foo() { console.log(value); } function bar() { var value = 2; foo(); } bar(); / / 1Copy the code
Lexical scope means that when a function is defined, its scope is already defined, regardless of where it is executed. Therefore, lexical scope is also called “static scope”.
1.4.2 Dynamic scope
JavaScript doesn’t actually have dynamic scope. It has lexical scope and is straightforward. But this mechanism is somewhat like dynamic scope.
Var obj = {m: function m(){console.log(this.num)}, msunh: 97} var num = 1997; var n = obj.m; obj.m() // 97 n() //1997Copy the code
You should now be familiar with the concept of scope and how variables are assigned to scopes based on where and how they are declared. But there is a subtle connection between where variable declarations occur in a scope, and that is the promotion we are talking about.
1.5 improve
1.5.1 Variable promotion
a = 2;
var a;
console.log( a );
Copy the code
What do you feel in console.log(..) What will be printed in the statement?
Many developers would expect undefined because the statement var a appears after a = 2, which naturally looks like the variable has been redefined and thus given undefined by default. However, the output will be 2.
console.log( a );
var a = 2;
Copy the code
You might be tempted to think that because the previous code snippet showed a behavior that didn’t look top to bottom, maybe in this code snippet, 2 would be printed as well. Others argue that because variable A was used before it was declared, this must cause a ReferenceError to be thrown.
Unfortunately, neither guess is correct. The output is undefined
The compiler hits again
When you see var a = 2; You may think of this as a statement. But JavaScript actually thinks of these as two statements: var a; And a = 2; . The first statement, the declaration, is processed at compile time. The second statement, assignment, is left in place for the execution phase.
So our first code snippet should be thought of as being handled like this:
var a;
Copy the code
a = 2;
console.log( a );
Copy the code
Similarly, our second code snippet is actually handled as:
var a;
Copy the code
console.log( a );
a = 2;
Copy the code
So, a somewhat metaphorical way to think about this processing is that variable and function declarations are “moved” to the top of the code from where they appear in the code stream. This gave rise to the name “ascension.”
Note: == only the declaration itself is promoted; any assignment or other execution logic is left in place ==. It would be a disaster if promotion rearranged the executable logic of our code.
1.5.2 Function promotion
foo();
function foo() {
console.log( a ); // undefined
var a = 2;
}
Copy the code
The declaration of function foo (which in this case also contains an implied value that is actually a function) is promoted so that the call to the first line can be executed.
It is also important to note that promotion is scoped. So while our previous code snippet was simplified to contain only global scopes, the function foo(..) we are now examining is Var a is promoted to foo(..). “(obviously, not the top of the program). So the program might be more accurately interpreted as:
function foo() {
var a;
console.log( a ); // undefined
a = 2;
}
foo();
Copy the code
Note: Function declarations are enhanced, as we have seen. But function expressions don’t.
When duplicate declarations occur, functions take precedence
foo(); // 1
var foo;
function foo() {
console.log( 1 );
}
foo = function() {
console.log( 2 );
};
Copy the code
The 1 is printed instead of the 2! This code snippet is interpreted by the engine to execute as
function foo() {
console.log( 1 );
}
foo(); // 1
foo = function() {
console.log( 2 );
};
Copy the code
Because of the promotion behavior, ES6 proposes block-level scopes.
1.6 Block-level scope for ES6
Simply put, the curly braces {… } is the block-level scoped region.
if (true) { var a = 1; } console.log(a); / / the result???????Copy the code
When you run it, you’ll see that the result is still 1, and that the variable a defined and assigned in curly braces goes global. Suffice it to say that javascript does not natively support block-level scopes.
But the ES6 standard proposes to “create block-level scopes” by using let and const instead of the var keyword. That is, block-level scope works if the code above is changed as follows
if (true) {
let a = 1;
}
console.log(a); // ReferenceError
Copy the code
As long as the let command exists in the block-level scope, the variables it declares are “binding” to the region, no longer subject to external influence.
var tmp = 123;
if (true) {
tmp = 'abc'; // ReferenceError
let tmp;
}
Copy the code
==ES6 explicitly states that if there are let and const commands in a block, the variables declared by the block to those commands form a closed scope from the start. Any time these variables are used before declaration, an error is reported. = =
In short, the variable is not available within the code block until it is declared using the let command. This is grammatically known as a “temporal dead zone” (TDZ).
1.7 How do I Create a scope
In javascript, we have several ways to create/change scopes:
- Define functions and create functions (recommended) :
Function foo () {// create a function scope for foo}Copy the code
- Creating block-level scopes with lets and const (recommended) :
for (let i = 0; i < 5; i++) {
console.log(i);
}
console.log(i); // ReferenceError
Copy the code
- Try catch creation scope (not recommended),err only exists in the catch clause:
try { undefined(); // Force an exception} catch (err) {console.log(err); // TypeError: undefined is not a function } console.log( err ); // ReferenceError: `err` not foundCopy the code
-
Using eval to “cheat” lexical scopes (not recommended)
-
Use with to cheat lexical scope (not recommended) :
Eval is not recommended because the use of dynamically generated code in a program is rare because the benefits do not outweigh the performance costs. Eval (..) is not recommended. And with are affected by strict mode (constraints). With is completely disallowed, and eval(..) is used indirectly or unsafely, while preserving the core functionality. It was also banned.Copy the code
In summary, there are only two ways to create scopes: define function creation and let const creation.
2. The closure
2.1 What is a closure
There are various explanations and explanations for closures
A closure is a function that can remember and access its lexical scope, even when the function is executed outside its lexical scope.
I think it’s easy to understand.
Functions that can access variables inside other functions are called closures.
From the scope we learned in section 1 we know the following
- What makes the Javascript language special is that global variables can be read directly from inside functions.
var n=999; The function f1 () {alert (n); } f1(); / / 999Copy the code
- Local variables inside a function cannot be read outside the function.
The function f1 () {var n = 999; } alert(n); // errorCopy the code
How do I read local variables from the outside? For a variety of reasons, we sometimes need to get local variables inside a function. However, as mentioned earlier, this is not normally possible and can only be achieved through workarounds.
That is, inside the function, define another function.
The function f1 () {var n = 999; The function f2 () {alert (n); / / 999}}Copy the code
In the above code, the function f2 is included inside the function f1, and all local variables inside f1 are visible to F2. But the other way around, local variables inside F2 are invisible to F1. This is the Javascript language’s “chained scope” structure, where child objects look up all of the parent’s variables, level by level. Therefore, all variables of the parent object are visible to the child object, and vice versa.
F2 can read local variables in F1, so if f2 is returned, we can read internal variables outside f1. = =
The function f1 () {var n = 999; The function f2 () {alert (n); } return f2; } var result=f1(); result(); / / 999Copy the code
F2 function, is the closure!!
So, in essence, a closure is a bridge between the inside and outside of a function.
2.2 Functions of closures
- Protect the private variables of functions from external interference, and realize the privatization of methods and attributes.
- The value of a variable is always kept in memory.
How to understand these two points? Look at the code below.
Function f1(){var n = 999; NAdd = function(){function f2(){alert(n); } return f2; } var result = f1(); result(); / / 999 nAdd bring them any closer (); result(); / / 1000Copy the code
- In this code, result is actually the closure f2 function. It runs twice, the first with a value of 999 and the second with a value of 1000. This proves that the local variable n in function f1 is kept in memory and is not automatically cleared after f1 is called.
- Why is that? The reason is that F1 is the parent of F2, and F2 is assigned to a global variable, which results in F2 always being in memory, and f2’s existence depends on F1, so F1 is always in memory and not collected by garbage collection when the call ends.
2.3 Application of closures
Closures offer a great deal of freedom in how code is organized, and they have a wide range of uses in everyday use.
The simple ones are:
- Successful callback to ajax request
- Callback method for event binding
- The delay callback of setTimeout
- Function returns another anonymous function inside the function
Here are a few application scenarios to help us understand closures:
- Constructor’s private property
- Function throttling, anti – shake
(1) Since there is no natural class implementation in javascript, some private properties that do not want to be externally modified can be implemented using closures.
function Person(param) { var name = param.name; // Private attribute this.age = 18; This.sayname = function () {console.log(name); } this.getAge = functin(){ console.log(this.age) } } const tom = new Person({name: 'tom'}); tom.age += 1; // Common attributes, external can change Tom.getage (); // 19 tom.name = 'jerry'; Tom.sayname (); tom.sayname (); // tomCopy the code
(2) Function throttling, anti-shake (take anti-shake as an example)
function debounce(fn, delay) { var timer = null; return function () { var that = this; var args = arguments; clearTimeout(timer); Timer = setTimeout(function () {fn.apply(that, args); }, delay || 500); }; }Copy the code
The buffered callback uses a timer to record the time. Each time the callback is executed, the previous timer will be cleared. Note that the timer is a closure variable that always holds the previous timer.
2.4 Test the sword
Can you answer seven closure interview questions