Home > Web Front-end > JS Tutorial > Detailed introduction to the various methods of JavaScript inheritance and their advantages and disadvantages

Detailed introduction to the various methods of JavaScript inheritance and their advantages and disadvantages

黄舟
Release: 2017-05-14 10:12:02
Original
1284 people have browsed it

This article mainly introduces the various ways and advantages and disadvantages of in-depth understanding of JavaScript inheritance. It has certain reference value. Interested friends can refer to it

Written in front

This article explains the various inheritance methods and advantages and disadvantages of JavaScript.

Note:

is the same as "JavaScript In-depth Creating Objects", more like notes.

Hey, let me sigh again: "JavaScript Advanced Programming" is really well written!

1. Prototype chain inheritance

function Parent () {
  this.name = 'kevin';
}

Parent.prototype.getName = function () {
  console.log(this.name);
}

function Child () {

}

Child.prototype = new Parent();

var child1 = new Child();

console.log(child1.getName()) // kevin
Copy after login

Problem:

1.Referenceproperty of type Shared by all instances, for example:

function Parent () {
  this.names = ['kevin', 'daisy'];
}

function Child () {

}

Child.prototype = new Parent();

var child1 = new Child();

child1.names.push('yayu');

console.log(child1.names); // ["kevin", "daisy", "yayu"]

var child2 = new Child();

console.log(child2.names); // ["kevin", "daisy", "yayu"]
Copy after login

2. When creating an instance of Child, parameters cannot be passed to Parent

2. Borrow the constructor (Classic inheritance)

function Parent () {
  this.names = ['kevin', 'daisy'];
}

function Child () {
  Parent.call(this);
}

var child1 = new Child();

child1.names.push('yayu');

console.log(child1.names); // ["kevin", "daisy", "yayu"]

var child2 = new Child();

console.log(child2.names); // ["kevin", "daisy"]
Copy after login

Advantages:

1. Avoid reference type properties being shared by all instances

2. Can be used in Passing parameters to Parent in Child

For example:

function Parent (name) {
  this.name = name;
}

function Child (name) {
  Parent.call(this, name);
}

var child1 = new Child('kevin');

console.log(child1.name); // kevin

var child2 = new Child('daisy');

console.log(child2.name); // daisy
Copy after login

Disadvantages:

Methods are all defined in the constructor, and each time an instance is created, the method will be created.

3. Combination inheritance

Prototype chain inheritance and classic inheritance are two swords in one.

function Parent (name) {
  this.name = name;
  this.colors = ['red', 'blue', 'green'];
}

Parent.prototype.getName = function () {
  console.log(this.name)
}

function Child (name, age) {

  Parent.call(this, name);
  
  this.age = age;

}

Child.prototype = new Parent();

var child1 = new Child('kevin', '18');

child1.colors.push('black');

console.log(child1.name); // kevin
console.log(child1.age); // 18
console.log(child1.colors); // ["red", "blue", "green", "black"]

var child2 = new Child('daisy', '20');

console.log(child2.name); // daisy
console.log(child2.age); // 20
console.log(child2.colors); // ["red", "blue", "green"]
Copy after login

Advantages: Combining the advantages of prototype chain inheritance and constructors, it is the most commonly used inheritance pattern in JavaScript.

4. Prototypal inheritance

function createObj(o) {
  function F(){}
  F.prototype = o;
  return new F();
}
Copy after login

is the simulation implementation of ES5 Object.create, using the passed in object as the prototype of the created object .

Disadvantages:

Attribute values ​​containing reference types will always share the corresponding value, which is the same as prototype chain inheritance.

var person = {
  name: 'kevin',
  friends: ['daisy', 'kelly']
}

var person1 = createObj(person);
var person2 = createObj(person);

person1.name = 'person1';
console.log(person2.name); // kevin

person1.firends.push('taylor');
console.log(person2.friends); // ["daisy", "kelly", "taylor"]
Copy after login

Note: After modifying the value of person1.name, the value of person2.name has not changed, not because of person1 and person2 has an independent name value, but because person1.name = 'person1', the name value is added to person1, not the name value on the prototype is modified. .

5. Parasitic inheritance

Create a function that is only used to encapsulate the inheritance process. This function is used to enhance the object in some form internally. Finally the object is returned.

function createObj (o) {
  var clone = object.create(o);
  clone.sayName = function () {
    console.log('hi');
  }
  return clone;
}
Copy after login

Disadvantages: Like the borrowed constructor pattern, a method will be created every time an object is created.

6. Parasitic combined inheritance

For the convenience of everyone’s reading, the code of combined inheritance is repeated here:

function Parent (name) {
  this.name = name;
  this.colors = ['red', 'blue', 'green'];
}

Parent.prototype.getName = function () {
  console.log(this.name)
}

function Child (name, age) {
  Parent.call(this, name);
  this.age = age;
}

Child.prototype = new Parent();

var child1 = new Child('kevin', '18');

console.log(child1)
Copy after login

Combined inheritance The biggest disadvantage is that the parent constructor is called twice.

Once when setting the prototype of a subtype instance:

Child.prototype = new Parent();
Copy after login

Once when creating a subtype instance:

var child1 = new Child('kevin', '18');
Copy after login

Recall the simulation implementation of new, in fact, here In the sentence, we will execute:

Parent.call(this, name);
Copy after login

Here, we will call the Parent constructor again.

So, in this example, if we print the child1 object, we will find that both Child.prototype and child1 have an attribute called colors, and the attribute values ​​are ['red', 'blue', 'green'] .

So how can we keep improving and avoid repeated calls this time?

What if we don’t use Child.prototype = new Parent(), but indirectly let Child.prototype access Parent.prototype?

Let’s see how to implement it:

function Parent (name) {
  this.name = name;
  this.colors = ['red', 'blue', 'green'];
}

Parent.prototype.getName = function () {
  console.log(this.name)
}

function Child (name, age) {
  Parent.call(this, name);
  this.age = age;
}

// 关键的三步
var F = function () {};

F.prototype = Parent.prototype;

Child.prototype = new F();


var child1 = new Child('kevin', '18');

console.log(child1);
Copy after login

Finally, let’s encapsulate this inheritance method:

function object(o) {
  function F() {}
  F.prototype = o;
  return new F();
}

function prototype(child, parent) {
  var prototype = object(parent.prototype);
  prototype.constructor = child;
  child.prototype = prototype;
}

// 当我们使用的时候:
prototype(Child, Parent);
Copy after login

To quote the praise of parasitic combined inheritance in "JavaScript Advanced Programming":

The high efficiency of this method reflects that it only calls the Parent constructor once, and therefore avoids creating unnecessary and redundant properties on Parent.prototype. At the same time, the prototype chain remains unchanged; therefore, instanceof and isPrototypeOf can still be used normally. Developers generally believe that parasitic compositional inheritance is the most ideal inheritance paradigm for reference types.

The above is the detailed content of Detailed introduction to the various methods of JavaScript inheritance and their advantages and disadvantages. For more information, please follow other related articles on the PHP Chinese website!

source:php.cn
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template