Preface: It’s still an introductory article. There are several very important language features in Javascript - objects, prototype inheritance, and closures. Among them, closures are a new language feature for programmers who use the traditional static language C/C++. This article will start with examples to introduce the language features of Javascript closures, and combine a little ECMAScript language specifications to enable readers to understand closures more deeply.
Note: This article is an introductory article, and the example materials are compiled on the Internet. If you are a master, you are welcome to put forward technical suggestions and opinions on the article. This article discusses Javascript, and you don’t want to compare languages. If you are naturally discomfortable with Javascript, please take a detour.
What is a closure
What is a closure? Closure is Closure, which is a new feature that static languages do not have. But closures are not something that is so complicated that they are incomprehensible. In short, closures are:
Closures are the set of local variables of the function, but these local variables will continue to exist after the function returns.
Closures are the "stack" of the function and are not released after the function returns. We can also understand that these function stacks are not allocated on the stack but are allocated on the heap.
When defining another function within a function, a closure will be generated
The second definition above is the first supplementary description, which extracts the subject-predicate object of the first definition - the closure is the set of 'local variables' of the function. It's just that this local variable can be accessed after the function returns. (This is not an official definition, but this definition should be more conducive to your understanding of closures)
As local variables, they can be accessed by the code in the function, and there is no difference between this and static languages. The difference between closures is that local variables can still be accessed by code outside the function after the function is executed. This means that the function must return a "reference" to the closure, or assign this "reference" to an external variable to ensure that local variables in the closure are accessed by external code. Of course, the entity containing this reference should be an object, because in Javascript, all the rest except the basic types are objects. Unfortunately, ECMAScript does not provide relevant members and methods to access local variables in the closure. However, in ECMAScript, the inner function defined in the function object is a local variable that can directly access external functions. Through this mechanism, we can complete access to the closure in the following way.
function greeting(name) {
var text = 'Hello ' + name; // local variable
// Each time a closure is generated and the internal function object is returned to the caller
return function() { alert(text); }
}
var sayHello=greeting("Closure");
saysHello() // Access the local variable text through the closure
The execution result of the above code is: Hello Closure, because after the greeting function is executed, the sayHello() function can still access the local variable text defined within it.
OK, this is the effect of the legendary closure. Closures have many application scenarios and modes in Javascript, such as Singleton, Power Constructor, and other Javascript modes that are inseparable from the use of closures.
ECMAScript closure model
How does ECMAScript implement closures? If you want to have an in-depth understanding, you can obtain the ECMAScript specifications for research. I will only give a simple explanation here, and the content also comes from the Internet.
When the function of the ECMAscript script runs, each function association has an execution context scenario (Execution Context), which contains three parts.
The LexicalEnvironment
The VariableEnvironment
This binding
The third point of this binding has nothing to do with closures and is not discussed in this article. A variable identifier used in the grammatical environment for parsing the function execution process. We can think of a grammatical environment as an object that contains two important components, the environment record (Enviroment Recode), and an external reference (pointer). The environment record contains local variables and parameter variables declared internally by the function, and the external reference points to the context execution scenario of the external function object. Global context This reference value is NULL in the scenario. Such a data structure forms a one-way linked list, each reference points to the outer context scenario.
For example, the closure model in our example above should be like this. The sayHello function is at the lowest level, the upper level is the function greeting, and the outermost level is the global scene. As shown in the figure below: Therefore, when saysHello is called, saysHello will find the value of the local variable text through the context scene, so the "Hello Closure" variable environment (The VariableEnvironment) and the grammar environment are basically the same. For specific differences, please refer to the ECMAScript specification document.
Sample column of closures
In the previous article, I roughly understand what Javascript closure is and how closures are implemented in Javascript. Below we will help you understand closures more deeply by targeting some examples. There are 5 examples below, and the examples are from JavaScript Closures For Dummies (mirror). Example 1: Local variables in closures are references rather than copies
function say667() {
// Local variable that ends up within closure
var num = 666;
var sayAlert = function() { alert(num); }
num++;
return sayAlert;
}
var sayAlert = say667();
sayAlert()
Therefore, the execution result should pop up 667 instead of 666.
Example 2: Multiple functions bind the same closure because they are defined within the same function.
function setupSomeGlobals() {
// Local variable that ends up within closure
var num = 666;
// Store some references to functions as global variables
gAlertNumber = function() { alert(num); }
gIncreaseNumber = function() { num++; }
gSetNumber = function(x) { num = x; }
}
setupSomeGlobals(); // Assign values to three global variables
gAlertNumber(); //666
gIncreaseNumber();
gAlertNumber(); // 667
gSetNumber(12);//
gAlertNumber();//12
Example 3: When assigning functions in a loop, these functions will bind the same closure
function buildList(list) {
var result = [];
for (var i = 0; i < ; i++) {
var item = 'item' + list[i];
( function() {alert(item + ' ' + list[i])} );
}
return result;
}
function testList() {
var fnlist = buildList([1,2,3]);
// using j only to help prevent confusion - could use i
for (var j = 0; j < ; j++) {
fnlist[j]();
}
}
The execution result of testList is that the item3 undefined window pops up three times, because these three functions bind the same closure, and the item's value is the last calculated result, but when i jumps out of the loop, the i value is 4, so the result of list[4] is undefined.
Example 4: All local variables of external functions are in the closure, even if this variable is declared after the internal function definition.
function sayAlice() {
var sayAlert = function() { alert(alice); }
// Local variable that ends up within closure
var alice = 'Hello Alice';
return sayAlert;
}
var helloAlice=sayAlice();
helloAlice();
The execution result is a window with "Hello Alice" pop-up. Even if the local variable declares after the function saysAlert, the local variable can still be accessed.
Example 5: Create a new closure every time the function is called
function newClosure(someNum, someRef) {
// Local variables that end up within closure
var num = someNum;
var anArray = [1,2,3];
var ref = someRef;
return function(x) {
num += x;
(num);
alert('num: ' + num +
'\nanArray ' + () +
'\ ' + );
}
}
closure1=newClosure(40,{someVar:'closure 1'});
closure2=newClosure(1000,{someVar:'closure 2'});
closure1(5); // num:45 anArray[1,2,3,45] ref:'someVar closure1'
closure2(-10);// num:990 anArray[1,2,3,990] ref:'someVar closure2'
Application of closures
Singleton single piece:
var singleton = function () {
var privateVariable;
function privateFunction(x) {
...privateVariable...
}
return {
firstMethod: function (a, b) {
...privateVariable...
},
secondMethod: function (c) {
...privateFunction()...
}
};
}();
This single piece is achieved through a closure. Encapsulation of private members and methods is completed through closures. Anonymous main function returns an object. The object contains two methods. Method 1 can use private variables and Method 2 can access internal private functions. The thing to note is the '()' where the anonymous main function ends. Without this '()', a single piece cannot be generated. Because anonymous functions can only return unique objects and cannot be called elsewhere. This is the method of using closures to generate single pieces.