Closures are an important feature of JS that must be mastered by the front end. This article explains the features of closures in conjunction with some of the JS mechanisms you’ve covered before.
Explain the closure
A closure must consist of at least two functions, one of which, upon entering the execution context, if an inner function is defined, accesses the variable object of the function in the execution context. A closure is formed. Here’s an example:
function A(){ / / A function
var n1 = 10;
var n2 = 20;
function B(){ / / B function
return n1 + n2;
}
return B;
}
var n = A();
console.log(n()); / / 30
Copy the code
- In the above code, there is A function A defined in the global context, and A function B defined inside.
- A returns B, but B is not called (B is cached in the variable n).
- The Closure is generated when B (n()) accesses the value of the variable object in A.
Note that most reference books and technical articles consider B functions to be closures. But in Chrome’s Scope you’ll see that the Closure points to the A function.
To form A closure, there must be at least two functions, A and B. This article takes Chrome as the standard and recognizes A function as A closure.
The JS garbage collection mechanism was introduced in the previous series of memory mechanisms. When a variable loses reference in memory, the JS memory collection mechanism algorithm will find the variable and reclaim, freeing its memory.
The execution context describes that when the function lifecycle ends after execution, the function context loses reference. The occupied memory space will be freed by the garbage collector. The formation of closures prevents this process.
Look at this example:
var B;
function A() {
var num = 2;
B = function() {
console.log(++num);
}
}
A();
B(); / / 3
B(); / / 4
Copy the code
- After calling A function, the anonymous function in A live object is cached in the global variable B
- B holds the reference to the context of A’s function and the variable object of A
- Because A’s variable object is stored in B, A call to B will self-add and then print the variable num in the context of A
Actual usage scenarios for closures
Implementing private variables
If you’ve looked at the source code for jQuery, you know that the entire code is wrapped up in an IIFE. Mount the object you want to expose outside the function on the window to implement encapsulation.
(function(window.undefined) {
var jQuery = function() {}
// ...
window.jQuery = window.$ = jQuery; }) (window);
Copy the code
- The immediate execution function (IIFE) is passed with the window object. The first parameter is also passed with the window object, and the second parameter is really undefined
- JQuery function expressions are defined internally
- This exposure is achieved by mounting jQuery to the jQuery variable of the Window object
- Window.$is an alias, used for shorthand.
We can write a similar implementation like this, finding the area of a circle in terms of its radius:
(function(window){
var R = 3;
function getCircleArea(r){
var r = r || R;
var s = r * r * Math.PI;
return s.toFixed(2);
}
window.getCircleArea = getCircleArea; }) (window);
// console.log( R ); // Error R is not defined
console.log( getCircleArea() ); / / "28.27"
console.log( getCircleArea(4));/ / "50.27"
Copy the code
- The immediate execution function (IIFE) can be thought of as a closure that exposes only one method, getCircleArea
- R can be regarded as a private variable of the immediate-execution function and is not accessible outside of the immediate-execution function
- Call the exposed getCircleArea function, and if the radius parameter is not passed, the result will be calculated by default as a private variable R
- GetCircleArea If passed, the area is calculated based on the value passed
Function Currying
The function Currying is used to transform a function that takes multiple arguments into a calculation that takes a single argument
function add(a, b) {
return a + b;
}
console.log(add(1.2)); / / 3
Copy the code
So if I curify it
function add(a) {
return function(b){
returna + b; }}console.log(add(1) (2)); / / 3
Copy the code
In code, function currying is a closure of the implementation, it is more practical use of function anti – shaking, interception, and so on, function currying will be introduced in the following article.
Functional Programming
Higher-order functions in functional programming come in two forms:
- Function is passed as an argument
- Function is output as a return value
The second scenario in which a function is output as a return value uses a closure example:
function isType(type) {
return function (obj) {
returnobj.constructor === type; }}const isArray = isType(Array);
const isObject = isType(Object);
const isFunction = isType(Function);
console.log(isArray([])); // true
console.log(isObject({})); // true
console.log(isFunction(function(){})); // true
Copy the code
The anonymous function returned by a higher-order isType(i.e. closure) compares the constructor of the parameter object to the type passed in by the higher-order function. Calls to higher-order functions cache the specific corresponding types. This is where closures come into play.
Pay attention to memory loss
Last but not least, when you use closures, the closure context is off the stack and the variable objects inside are stored in memory, and there is a certain performance cost. Use JS features such as closures appropriately in the context of your actual usage scenarios.