preface
Javascript you Don’t Know is a must-read series on front-end learning that allows casual javascript developers to dive into the guts of the language and understand the purpose of every single component of javascript. This book introduces two topics in the series: “scopes and closures” and “This and object prototypes.” These two are also worth us to learn and ponder over the two are just the content, today we use the way of mind mapping to read closely. (Mind map images may be a little small, but click on them and you’ll learn something.)
The first part is scopes and closures
What is the scope
A scope is a set of rules for determining where and how to look for variables (identifiers). If the purpose of the lookup is to assign a value to a variable, an LHS query is used; If the goal is to get the value of a variable, RHS queries are used. The assignment operator causes an LHS query. The assignment operation of. The = operator or an operation passing in an argument when calling a function results in an assignment of the associative scope. The JavaScript engine first compiles the code before it executes, during which a sound like var a = 2 is broken down into two separate steps:
- First, var a declares new variables in its scope. This happens at the very beginning, before the code executes.
- Next, variable A = 2 is queried (LHS query) and assigned to it.
LHS and RHS query will be at the beginning of the currently executing scope, if necessary (which means that they did not find that need the identifier), to superior scope will continue to find the target identifier, so each time to the next level scope (floor), finally arrived in the global scope (top), no matter find or not find will stop.
Unsuccessful RHS references cause ReferenceError exceptions to be thrown. Unsuccessful LHS references result in the automatic implicit creation of a global variable (in non-strict mode) that uses the target of the LHS reference as an identifier, or a ReferenceError exception thrown (in strict mode).
Lexical scope
Lexical scope means that the scope is determined by the position of the function declaration when the code is written. The lexical analysis phase of compilation basically knows where and how all identifiers are declared, and thus can predict how they will be looked up during execution.
There are two mechanisms in JavaScript that can “cheat” lexical scopes: eval(..) And with. The former allows you to evaluate a string of “code” containing one or more declarations to modify existing lexical scopes (at run time). The latter essentially creates a new lexical scope (again at run time) by treating a reference to an object as a scope and an object’s attributes as identifiers in the use domain.
The side effect of these two mechanisms is that the engine cannot optimize scoped look-ups at compile time, because the engine can only discreetly assume that such optimizations are not valid. Using any of these mechanisms will cause your code to run slowly. Don’t use them.
Function scope and block scope
Functions are the most common unit of scope in JavaScript. Essentially, variables or functions declared inside a function are “hidden” in the scope at which they are located, which is an intentional design principle of good software.
But functions are not the only unit of scope. Block scoping means that variables and functions can belong not only to the scope they are in, but also to a code block (usually {.. } inside).
As of ES3, try/catch structures have block scope in catch clauses. The let keyword (a cousin of the var keyword) was introduced in ES6 to declare variables in arbitrary blocks of code. if(..) { let a = 2; } would declare a {.. } block, and add variables to the block.
Some people think that block scope should not be used entirely as an alternative to function scope. Both functions should exist at the same time, and developers can and should choose which scope to use as needed to create good code that is readable and maintainable.
ascension
We’re used to var a = 2; As a statement, which the JavaScript engine does not. It treats var a and a = 2 as two separate declarations, the first for the compile phase and the second for the execution phase.
This means that no matter where the declaration in scope appears, it will be processed first before the code itself is executed. This process can be visualized as all declarations (variables and functions) are “moved” to the top of their scope. This process is called promotion.
The declaration itself is promoted, but assignment operations, including assignment of function expressions, are not.
Be careful to avoid duplicate declarations, especially when normal var declarations are mixed with function declarations, which can lead to dangerous problems!
Scope closures
Closures are like a mysterious, uncivilized world separated from JavaScript that only the bravest can reach. But it’s really just a standard, obviously about how to write code in a lexical environment where functions are passed as values on demand.
Closures occur when a function can remember and access its lexical scope, even if the function is executed outside the current lexical scope.
If you don’t recognize closures and understand how they work, it’s easy to make mistakes when using them, such as loops. But closures are also a very powerful tool for implementing patterns such as modules in many forms. Modules have two main features:
(1) A wrapper function is called to create the inner scope; (2) The return value of the wrapper function must include at least one reference to the inner function, so that a closure covering the entire inner scope of the wrapper function is created.
Now we see closures everywhere in our code, and we can identify closures and do something useful with them!
Part two: This and object prototypes
A. this B. this C. this D. this
If you want to determine the this binding of a running function, you need to find where the function is called directly. Once found, we can apply the following four rules in sequence to determine the object bound to this.
-
Called by new? Bind to the newly created object.
-
Called by call or apply (or bind)? Binds to the specified object.
-
Called by a context object? Bind to that context object.
-
Default: bind to undefined in strict mode, otherwise bind to global object.
It is important to note that some calls may inadvertently use the default binding rules. If you want to “safely” ignore this binding, you can use a DMZ Object such as ø = object.create (null) to protect global objects. Instead of using the four standard binding rules, the arrow function in ES6 determines this based on the current lexical scope. Specifically, the arrow function inherits the this binding of the outer function call (whatever this is bound to). This is actually the same mechanism as self = this in previous ES6 code.
object
Objects in JavaScript have a literal form (e.g. Var a = {.. }) and constructs (e.g. Var a = new Array(..) ). The literal form is more commonly used, but sometimes the constructed form can provide more options.
Many people think that “everything in JavaScript is an object”, which is wrong. The object is one of six (or seven, depending on your point of view) base types. Objects have subtypes, including function. Different subtypes have different behaviors. For example, the inner tag [object Array] indicates that this is an Array of subtypes of the object.
An object is a collection of key/value pairs. Attribute values can be obtained using either the.propName or [“propName”] syntax. When a property is queried, the engine actually calls the internal default [[Get]] operation ([[Put]] when setting the property value). The [[Get]] operation checks to see if the object itself contains the property, and if it doesn’t, looks for the [[Prototype]] chain (see Chapter 5).
The properties of a property can be controlled by a property descriptor, such as Writable and 64x. Alternatively, use object.preventExtensions (..) And the Object. Seal (..) And the Object. Freeze (..) To set the immutability level of an object (and its properties).
Properties do not necessarily contain values — they may be “access descriptors” with getters/setters. In addition, properties can be enumerable or non-enumerable, which determines whether they appear in for.. In loop.
You can use ES6’s for.. The of syntax iterates over values in data structures (arrays, objects, and so on), for.. Of looks for a built-in or custom @@iterator and calls its next() method to iterate over the data values.
Mixed object “class”
Classes are a design pattern. Many languages provide native syntax for class-oriented software design. JavaScript has a similar syntax, but it’s completely different from classes in other languages.
Class means copy.
When a traditional class is instantiated, its behavior is copied into the instance. When a class is inherited, its behavior is also copied into its subclasses.
Polymorphisms (functions with the same name but different functions at different levels of the inheritance chain) appear to refer to the parent class from the subclass, but in essence refer to the result of replication.
JavaScript does not automatically create copies of objects (like classes do).
Mixin patterns (explicit or implicit) can be used to simulate the copying behavior of classes, but often result in ugly and brittle syntax such as explicit pseudo-polymorphisms (otherobj.methodName.call (this,…)). ), which makes the code harder to understand and maintain.
Furthermore, explicit mixing can’t actually completely mimic the copying behavior of a class, because objects (and functions! Remember that functions are also objects.) Only references can be copied, not the referenced object or the function itself. Ignoring this point can lead to many problems.
In general, emulating classes in JavaScript is not worth the cost, and while it may solve the current problem, it may create more pitfalls.
The prototype
To access a property that doesn’t exist in the object, the [[Get]] operation (see Chapter 3) looks for the object associated with [[Prototype]] inside the object. This association actually defines a “stereotype chain” (sort of like a nested scope chain) that is iterated over when attributes are found.
All normal objects have built-in Object.prototype that points to the top of the prototype chain (such as the global scope) and stops if the specified property is not found in the prototype chain. ToString (), valueOf(), and other generic functions are present on the Object.prototype Object, so all objects in the language can use them.
The most common way to associate two objects is to make a function call using the new keyword. In the chapter of the call), a new object is created that is associated with another object. The four steps (Chapter 2) create a new object associated with other objects.
Calling a function with new associates the.prototype property of the new object with “other objects.” Function calls with new are often referred to as “constructor calls,” although they are not really the same as class constructors in traditional class-oriented languages.
One of the core differences of the mechanics in JavaScript is that there is no copying, objects are internal to each other
Although these mechanisms are very similar to classinitialization and class inheritance in traditional class-oriented languages, javascript mechanisms are very similar to classinitialization and class inheritance in traditional object-oriented languages with one central difference, which is that they don’t copy, Objects are related via an internal [[Prototype]] chain.
For a variety of reasons, terms that end in “inheritance” (including “archetypal inheritance”) and other object-oriented terms don’t help you understand the real mechanics of JavaScript (and not just limit our thinking patterns).
By contrast, “delegate” is a more appropriate term because the relationship between objects is not replication but delegation.
Behavior to entrust
You can choose whether to use class and inheritance design patterns in software architecture. Most developers take it for granted that classes are the only (appropriate) way to organize code, but in this chapter we’ve seen another, rarer but more powerful design pattern: behavioral delegation.
Behavioral delegation considers objects to be siblings, delegating to each other, rather than superclass and subclass. JavaScript’s [[Prototype]] mechanism is essentially a behavior delegation mechanism. That said, we can either try to implement the class mechanism in JavaScript (see Chapters 4 and 5) or embrace the more natural [[Prototype]] delegate mechanism.
When you design code using only objects, you not only make the syntax cleaner, but you also make the code structure clearer.
Object association (before objects are related to each other) is a coding style that advocates creating and associating objects directly without abstracting them into classes. Object associations can be implemented quite naturally with behavior delegates based on [[Prototype]].
extension
Mind map can clearly restore the knowledge structure system of the whole book. If you have not read the book, you can quickly preview it according to the idea of the mind map to improve learning efficiency. Learning new things is always easy to forget, I prefer to use mind maps to make records when reading books, so as to facilitate their later review, if you have read this book, I also suggest you to collect review. If you have any suggestions or ideas, please leave a message or add me to wechat: 646321933
Javascript you don’t know Part 2 of Volume 1 online documentation
Javascript you don’t know (volume 1) PDF download address
Close reading volume 2 of javascript You Don’t Know
Close reading of Illustrated HTTP
Mind map download address