Prototype and prototype chain
(Current personal understanding, only for reference)
First, the way to call the function
Know what a prototype and a prototype chain are. First of all, it is necessary to know how functions are called. All knowledge points should not exist in isolation. Only with a comprehensive understanding of function calls can we understand what happens in the process of prototype chain generation. So let me just make a couple of statements here.
1.1 Direct call mode
function fn() {}
fn()
const a = {
name: 'mother zha'.fn: function() {
console.log(this.name)
}
}
a.fn() / / mama zha
Copy the code
That’s the way most people learn functions first. Fn.apply (null) is the default method to call a function. Just hide it. Apply (null).
However, I did not find the answer in MDN, I do not know whether it is so, just for reference.
Functions in objects are also conventionally called methods.
When calling a function directly, the default is not to explicitly return a value. It will return you a base type undefined by default.
This points to Windows global
1.2 the call and apply
function Test1() {
this.name = Grasshopper essence
}
function Test2() {
Test1.apply(this)}const test = new Test2()
console.log(test) / / the grasshopper
Copy the code
The difference between pointing to call directly on this and applying is that it is followed by an array.
function Test1(name, color) {
this.name = name
this.color = color
}
function Test2() {
Test1.apply(this.'grasshopper'.'red')
Test1.call(this['grasshopper'.'red'])}Copy the code
There is a rumor that the call method is more efficient than apply. The following is the result of others’ practice.
· Issue #6 · Noneven /__ (github.com)
1.3 Construct the call method
This is how we call the prototype chain, the constructor process: create this at the top of the inner function, and return it.
The main difference is that This** points to. Barring this, there should be only one way to call the ** function.
Second, the prototype
To learn about prototypes and prototype chains, you also need to be exposed to these terms:
Prototype __proto__ constructor
-
Prototype: constructor instantiation returns the property. ** belongs to constructor instantiation, not constructor **
-
Protp: constructor instantiation process, the return of a property of this, the value of this attribute is the prototype, it is equivalent to a pointer name, can be called a, b, bb; Opening reality in browser controller is [[prototype]].
-
Constructor: Points to the constructor itself, with the emphasis that the property can be changed
function Person() {
this.name = 'the sheep sheep'
}
Person.prototype = {
age: 26
}
const girl = new Person()
console.log(girl.age) / / 26
Copy the code
Prototype. It is an object, so it has all the properties of a reference type.
The value of the age property was successfully printed. Prototype also uses some mysterious method to hang on top of this in constructor instantiation.
This mysterious method, the front end can not understand
function Person() {
var this = {
name: 'the sheep sheep'.__proto__: Person.prototype
}
return this
}
const girl = new Person()
Copy the code
In the Person constructor, this is generated during instantiation.
If.age is not found in the this. first layer, it goes to __proto__ by some method that doesn’t need the front end to know.
I don’t know when many textbook and video examples of prototypes started. The console prints __proto__, but I now print [[prototype]].
It is important to note that stereotypes are mounted on instances, not constructors
fucntino Car() {}
Car.prototype = {
color: 'white'
}
cons a = new car()
console.log(a.color) // 'white'
car.prototype.color = 'red;
console.log(a.color) // 'red'
Copy the code
Changing the values directly on the prototype will cause the instance properties to change, after all, you are only changing the values.
But here’s where the problem arises
functino Car() {}
Car.prototype = {
color: 'white'
}
const a = new Car()
Car.prototype = {
color: 'red'
}
console.log(a.color) // white
Copy the code
Confusion points in prototype assignment
The above example directly indicates that the value of [[prototype]] on object A is the prototype. The important thing is that this process is not a replication process.
It’s a relational process. During creation, or new, both a and the internal __proto__ are attached to Car. Prototype.
This is from Javascript you Don’t Know, volume 5.2.3
Because of this, the following code holds:
function Car() {
this.color = 'red'
}
Car.prototype = {
age: 18.constructor: Motorcycle
}
let car = new Car()
car.__proto__.age = 30
console.log(car) // [[prototype]] has an age of 30
Copy the code
The age on the prototype has been changed. This, in turn, causes the same variable that instantiates Car to change.
let car2 = new Car()
console.log(car2.age) / / 30
Copy the code
Is it not the case that the son beats the father and the younger suffers?
Some say archetypes exist for inheritance. So we have to block out the side effects of the shared properties of the prototype.
Prototype chain
function Car() {
this.name = 'one'
}
Car.prototype = {
color = 'red'
}
const car = new Car()
Copy the code
All the way down on the left is the prototype chain.
console.log(car.color) // 'red'
Copy the code
Properties that cannot be found on an object will be searched on the previous prototype until the location is found, or undefined.
Object. Prototype (car) {Object. Prototype (car) {Object.
__proto is a concrete representation attribute of [[Prototype]]. So [[Prototype]] represents __proto
Something seems wrong. There is no Function constructor.
This is because instantiation is an Object, so there is no Functino on the prototype chain; it refers directly to Object. The correct prototype chain diagram should look like this:
So this makes sense.
var F = function() {}; Object.prototype.a = function() { console.log('a'); }; Function.prototype.b = function() { console.log('b'); } var f = new F(); f.a(); // 'a' f.b(); // error: b = F.a(); // 'a' F.b(); // 'b'Copy the code