Closures and “this” are two fairly common topics, but have you ever thought that they’re both related to the same topic?
Welcome to our main character of this article, the execution context
Execution context
What is the execution context
It is easy to understand that the execution context is the environment in which JS code is executed. It consists of the following
ExecutionContextObj = {this: this, VO: variable object, scopeChain: closure related}Copy the code
Since JS is single-threaded, only one thing can happen at a time, and other things are queued in the specified context stack. When the JS interpreter initializes the execution code, it creates a global execution context on the stack, and then creates and pushes a new execution context stack with each function call. The execution context is popped after the function executes.
Five key points:
- Single thread
- Synchronous execution
- A global context
- Unrestricted function context
- Each function call creates a new context, including the call itself
Performs a context-established step
Create a stage
- Initialize the scope chain
- Creating a variable object
- Create the arguments
- Scanning function declaration
- Scanning variable declaration
- For this
Execution phase
- Initialize references to variables and functions
- Execute the code
this
When a function executes, this always refers to the object on which the function was called. To determine what this refers to, we need to determine which function this belongs to.
Point to the calling object
function foo() { console.log( this.a ); } var obj = { a: 2, foo: foo }; obj.foo(); / / 2Copy the code
Pointing to a global object
function foo() { console.log( this.a ); } var a = 2; foo(); / / 2Copy the code
Pay attention to
Var bar = foo a = 3 bar() // 3 not 2Copy the code
This example makes it clear that this is only determined when the function is called
Again a little bit around
function foo() { console.log( this.a ); } function doFoo(fn) { this.a = 4 fn(); } var obj = { a: 2, foo: foo }; var a =3 doFoo( obj.foo ); / / 4Copy the code
while
function foo() { this.a = 1 console.log( this.a ); } function doFoo(fn) { this.a = 4 fn(); } var obj = { a: 2, foo: foo }; var a =3 doFoo( obj.foo ); / / 1Copy the code
Why is that? Is it because a is read first in foo, similar to the principle of scope?
By printing this for foo and doFoo, we know that their this refers to window, and that their operations modify the value of a in window. The a set in foo is not read first
So if I change the code to
Function foo() {setTimeout(() => this.a = 1,0) console.log(this.a); } function doFoo(fn) { this.a = 4 fn(); } var obj = { a: 2, foo: foo }; var a =3 doFoo( obj.foo ); // 4 setTimeout(obj.foo,0) // 1Copy the code
The code results above confirm our guess.
The new construct points to the new object
A = 4 function a () {this.a = 3 this.calla = function() {console.log(this.a)}} a () // return undefined, a (). Var a = new a () a.calla () // 3,callA is stored in the object returned by new ACopy the code
apply/call/bind
function foo() { console.log( this.a ); } var obj = { a: 2 }; foo.call( obj ); // 2 //bind returns a new function function foo(something) {console.log(this.a, something); return this.a + something; } var obj = a: 2 }; var bar = foo.bind( obj ); var b = bar( 3 ); // 2 3 console.log( b ); / / 5Copy the code
Arrow function
The arrow function, in particular, does not have its own this; it uses the this value of the enclosing execution context (function or global).
var x=11; var obj={ x:22, say:()=>{ console.log(this.x); //this points to window}} obj.say(); Var obj2={x:22, say() {console.log(this.x); var obj2={x:22, say() {console.log(this.x); //this points to window}} obj2.say(); // 22 obj2.say.call({x:13}) // 13Copy the code
Event listener function
Points to the bound DOM element
Document. The body. The addEventListener (' click ', function () {the console. The log (this)}) / / click on the web / / < body >... </body>Copy the code
HTML
It is possible to write JS in an attribute of an HTML tag, in which case this refers to the HTML element.
<div id="foo" onclick="console.log(this);" ></div> <script type="text/javascript"> document.getElementById("foo").click(); //logs <div id="foo"... </script>Copy the code
The variable object
A variable object is an execution context-specific data scope that stores variable and function declarations defined in the context.
A variable object is an abstract concept that represents different objects in different contexts
The variable object of the global execution context
In the context of global execution, variable objects are global objects. In the top-level JS code, this refers to the global object, and the global variable is queried as an attribute of the object. In a browser, a window is a global object.
var a = 1
console.log(window.a) // 1
console.log(this.1) // 1Copy the code
The variable object of the function execution context
In the context of functions, the variable object VO is the active object AO.
When initialized, with the arguments attribute. The function code is executed in two phases
-
When the execution context is entered, the variable object includes
- parameter
- A function declaration that replaces an existing variable object
- Variable declarations that do not replace parameters and functions
-
Function performs
Modify the value of the variable object according to the code
For example
function test (a,c) { console.log(a, b, c, d) // 5 undefined [Function: c] undefined var b = 3; a = 4 function c () { } var d = function () { } console.log(a, b, c, d) // 4 3 [Function: c] [Function: D] var c = 5 console.log(a, b, c, d) // 4 3 5 [Function: d]} test(5,6)Copy the code
Let’s analyze the process
1. When creating the execution context
VO = {arguments: {0:5}, a: 5, b: undefined, c: [Function], }}}}}}}}}}}}}}}
- When executing code
As you can see from the final console, function declarations can be overwritten
The scope chain
First, look at the scope
scope
The accessibility of variables and functions controls the visibility and lifecycle of variables and functions. There are global scopes and local scopes.
Global scope:
Objects that can be accessed anywhere in the code have global scopes. There are several types:
-
Variables defined in the outermost layer;
-
Properties of a global object
-
Variables defined implicitly anywhere (variables that are not directly assigned) are defined implicitly everywhere in global scope, that is, variables that are not directly assigned through var declarations.
Local scope:
JavaScript scopes are defined by functions, and variables defined in a function are only visible inside that function, called function (local) scopes
The scope chain
A scope chain is a list of objects used to retrieve identifiers that appear in context code. Identifiers can be understood as variable names, parameters, and function declarations.
When a function is defined, it stores the parent variable object AO/VO in an internal property [[scope]], called the scope chain. Free variables are variables that are not declared inside a function. When a function needs to access a free variable, it looks up the data along the scope chain.
function foo() {
function bar() {
...
}
}
foo.[[scope]] = [
globalContext.VO
];
bar.[[scope]] = [
fooContext.AO,
globalContext.VO
];Copy the code
When the function runs the activation, it copies the [[scope]] property to create the scope chain, then creates the variable object VO, and adds it to the scope chain.
ExecutionContextObj :{VO:{}, scopeChain: [VO, [[scope]]]}Copy the code
closure
What is a closure
Closures, as defined by MDN, are functions that can access free variables. As mentioned earlier, free variables are variables that are not declared inside a function.
Closure form
function a() {
var num = 1
function b() {
console.log(num++)
}
return b
}
var c1 = a()
c1() // '1'
c1() // '2'
var c2 = a()
c2() // '1'
c2() // '2'Copy the code
Closure process
It’s not very rigorous. I may have omitted some process
- Run function A
- Create VO of function A, including variable num and function b
- When we define function b, we save a’s VO and global variable object into [[scope]]
- Return function B, save to c1
- Run the c1
- Creates a scope chain for C1 that holds the variable object VO of A
- Create VO for C1
- Run c1 to find the variable num, which does not exist in the VO of a, through the scope chain, find num stored in the VO of A, operate on it, and set num to 2
- Run c1 again and repeat step 2 with num set to 3
Some of the problems
From the above results, we can observe that the num variable accessed by C2 is not the same as the num variable accessed by c1. We can modify the code to confirm our guess
function a() {
var x = {y : 4}
function b() {
return x
}
return b
}
var c1 = a()
var c2 = a()
c1 === c2() // falseCopy the code
So we can be sure that the variables accessed by the closure are recreated independently of each other each time the parent function is run. Note that free variables created in the same function can be shared in different closures
function a() {
var x = 0
function b() {
console.log(x++)
}
function c() {
console.log(x++)
}
return {
b,
c
}
}
var r = a()
r.b() // 0
r.c() // 1Copy the code
The last
The article is quite long and covers a wide range. There may be a lot of mistakes. I hope you can correct them.
This post is part of the Advanced Front End series. Follow star on this blog or follow me on Github
reference
- Dig deeper into this in ES6 arrow functions
- You don’t know JS rolls up
- JavaScript deep execution context stack
- Understand JavaScript’s scope chain
- JavaScript deep variable objects
- Understanding JavaScript series 12: Variable Objects
- Understand the execution context of JavaScript