This article starts with an in-depth analysis and comparison of javascript inheritance implementations prior to the release of ES6, followed by an introduction to es6 support for class inheritance and a discussion of its advantages and disadvantages. Finally, “polymorphism”, which is rarely involved in javascript object-oriented programming, is introduced, and the idea of “operator overloading” is provided. This article assumes that you already know or understand the concepts of prototypes and prototype chains in JS.

Prior to ES6, javascript was not an object-oriented programming language per se, as it did not provide native support for encapsulation, inheritance, and polymorphism, the characteristics of object-oriented languages. However, it introduces the concept of prototype, which allows us to emulate classes in another way, and implements inheritance of shared attributes between parent and child classes and identification mechanism through prototype chain. In fact, the concept of object orientation is not a language feature per se, but a design concept. If you are well versed in object-oriented programming, you can write object-oriented code in a procedural language like C (typically Windows NT kernel implementations), and so can javascript! Because javascript itself does not have a language support standard for object-oriented programming, there is a bewildering array of “class inheritance” code. Fortunately, ES6 adds keywords like class, extends, and static to support object orientation at the language level, but that’s still a bit conservative! We’ll start with a list of common inheritance schemes prior to ES6, then take a look at es6’s class inheritance mechanism, and finally discuss javascript polymorphism.

Inheritance prior to ES6

Prototype assignment mode

In short, you assign an instance of the parent class directly to the prototype of the child class. The following is an example:

function Person(name){ this.name=name; this.className="person" } Person.prototype.getClassName=function(){ console.log(this.className) } function Man(){ } Man.prototype=new Person(); //1 //Man.prototype=new Person("Davin"); //2 var man=new Man; >man.getClassName() >"person" >man instanceof Person >trueCopy the code

As shown in 1 of the code, this method simply new an instance of the parent class and then assign it to the prototype of the child class. This is equivalent to assigning all method properties in the parent class prototype and all method properties hanging on this to the subclass prototype. Let’s look at man, which is an instance of man, because man doesn’t have a getClassName method, so we’ll go to the prototype chain, and we’ll find the getClassName of Person. The biggest problem with this approach is that subclasses cannot create private attributes from their parent classes. For example, if each Person has a name, we would initialize each Man with a different name, and then the subclass would pass that name to the parent class. For each Man, the name stored in the corresponding Person should be different, but this approach simply does not work. Therefore, this kind of inheritance is basically not needed in actual combat!

Call the constructor method

function Person(name){ this.name=name; this.className="person" } Person.prototype.getName=function(){ console.log(this.name) } function Man(name){ Person.apply(this,arguments) } var man1=new Man("Davin"); var man2=new Man("Jack"); >man1.name >"Davin" >man2.name >"Jack" >man1.getName() //1 error >man1 instanceof Person >trueCopy the code

The constructor of the subclass is called by this of the subclass instance, and the effect of inheriting the attributes of the parent class is achieved. In this way, each new instance of a subclass will have its own resource (name) after the constructor executes. However, this method only inherits the instance properties declared in the superclass constructor, but does not inherit the properties and methods of the superclass prototype, so the getName method cannot be found, so an error is reported at 1. In order to inherit both parent archetypes, combinatorial inheritance is born:

Combination of inheritance

function Person(name){ this.name=name||"default name"; //1 this.className="person" } Person.prototype.getName=function(){ console.log(this.name) } function Man(name){ Prototype = new Person(); var man1=new Man("Davin"); > man1.name >"Davin" > man1.getName() >"Davin"Copy the code

This is a simple example, which not only inherits the properties in the constructor, but also copies the properties in the superclass prototype chain. Man.prototype = new Person(); After this sentence is executed, the prototype of Man is as follows:

> Man.prototype
> {name: "default name", className: "person"}

Copy the code

That is, the Man prototype already has a name attribute, and the name passed to the constructor when man1 is created is redefined as a name attribute by this, which overwrites the name attribute of the prototype (the name of the prototype is still there), which is not elegant.

Separated combinatorial inheritance

This is currently the dominant inheritance method in ES5, with some people calling it “parasitic combination inheritance”. First of all, they’re the same thing. I came up with the name split combination inheritance, because it feels better not to be pussy-ass, and it’s more accurate. In summary, inheritance can be divided into two steps: constructor attribute inheritance and linking of subclass and superclass stereotypes. The so-called separation is in two steps; Composition means inheriting properties from both the subclass constructor and the stereotype.

function Person(name){ this.name=name; //1 this.className="person" } Person.prototype.getName=function(){ console.log(this.name) } function Man(name){ Prototype = object.create (person.prototype); var man1=new Man("Davin"); > man1.name >"Davin" > man1.getName() >"Davin"Copy the code

Here we use the object.creat (obj) method, which makes a shallow copy of the obj Object passed in. The main difference from the above composite inheritance is that the parent class stereotype is copied to the child class stereotype. The approach is clear:

  1. Constructor inherits the parent class attributes/methods and initializes the parent class.
  2. The subclass stereotype is associated with the superclass stereotype.

The constructor attribute () ¶

> Person.prototype.constructor
< Person(name){
   this.name=name; //1
   this.className="person" 
 }
> Man.prototype.constructor
< Person(name){
   this.name=name; //1
   this.className="person" 
  }

Copy the code

Constructor is the constructor of the class, and we see that both the Person and Man instances have constructor pointing to Person. Of course, this does not change the result of instanceof, but it is problematic for scenarios where construcor is needed. So we usually add something like:

Man.prototype.constructor = Man

Copy the code

Overall, under ES5, this approach is the preferred and in fact the most popular.

Writing at this point, the main inheritance way is introduced under the es5, before introducing es6 inheritance, we again look deep, here is the sole dry goods, we look at the Neat. A simplified in js source code:

/ / below for Neat source to simplify -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- the function Neat () call (this)} {Array. Neat. The prototype = Object. The create (Array) prototype) Neat. The prototype. The constructor = Neat -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- / var/test code neat=new Neat; > > neat. Push (1, 2, 3, 4) neat. Length / / 1 > neat [4] = 5 > neat. Length / / 2 > neat. The concat (/ June) / / 3Copy the code

Now, what does the split-wrapped code block above do?

A Neat object inherits from an array. Let’s take a look at the following test code and guess what the results of execution 1, 2, and 3 are. The desired results should be:

4, 5, 1,2,3,4,5,6,7,8Copy the code

But in fact:

4, 4 [[1,2,3,4], 6,7,8]Copy the code

God’s! It’s not scientific! why ?

The explanation I once read in an article by Ruan Yifeng is as follows:

Because subclasses can’t get the internal properties of the native constructor, either through array.apply () or by assigning them to the prototype object. The native constructor ignores this passed in by apply. That is, the this of the native constructor cannot be bound, resulting in no internal attributes. ES5 creates the instance object this of the subclass first, and then adds the attributes of the parent class to the subclass. Because the internal attributes of the parent class cannot be obtained, it cannot inherit the native constructor. For example, the Array constructor has an internal property [[DefineOwnProperty]] that is used to update the Length property when defining a new property. This internal property is not available in the subclass, causing the subclass’s Length property to behave improperly.

However, this is not the case! To be sure, it is not the native constructor that ignores this passed in by the apply method and causes the property to be unbound. Otherwise we wouldn’t have printed a 4 at 1. In addition, Neat can still call push and other methods normally, but some methods on the prototype are also problematic after inheritance, such as neat. Concat. In fact, we can also use array.call (this), such as the length property available. But why ask? Based on the symptoms, you can be sure that there is something wrong with this in the end, but what is it? Is there something we missed that is causing missing properties not to initialize properly? Or is it that the browser initializes arrays differently from custom objects? First, let’s look at the first possibility. The only possibility left out is the static method of the array (none of the above inheritance methods inherit from the parent static method). We can test this:

for(var i in  Array){
 console.log(i,"xx")
}

Copy the code

However, there is no line of output, which means Array has no static methods. Of course, this method can only iterate over enumerable properties. What if there are non-enumerable properties? Even if it does, it should be private in the browser’s eyes, and the browser doesn’t want you to do it! So the first case is pass. It could only have been the second case, and it wasn’t until ES6 came out that the answer was:

ES6 allows subclasses to be inherited from native constructors, because ES6 creates an instance object of the parent class, this, and then decorates this with the constructor of the subclass, making all behavior of the parent class inheritable.

Please note my bold words. “All” is a subtle word, not “none”, so the implication is that ES5 is part of it. Based on my previous tests (in ES5), subscripting and concat are problematic in Chrome, while most functions are fine. Of course, this may vary from browser to browser, This is probably the main reason jQuery returns the result set as a new extended array after each operation, rather than inheriting the array itself (and then returning this directly), since jQuery is compatible with all browsers. The problem facing Neat. Js is not so complicated, just bypassing the potholes. Without further ado, other objects in ES5 that, like arrays, browsers don’t let us have fun playing with are:

Boolean()
Number()
String()
Array()
Date()
Function()
RegExp()
Error()
Object()

Copy the code

Es6 inheritance mode

Es6 introduces class, extends, super, static(part of ES2016 standard)

class Person{ //static sCount=0 //1 constructor(name){ this.name=name; this.sCount++; GetName (){console.log(this.name)} static sTest(){console.log("static method test")}} class Man extends  Person{ constructor(name){ super(name)//3 this.sex="male" } } var man=new Man("Davin") man.getName() //man.sTest() Man.stest ()//4 The command output is Davin static method testCopy the code

ES6 explicitly states that there are only static methods inside a Class and no static properties, so 1 is problematic. ES7 has a proposal for static properties that Babel transcoders currently support. Those familiar with Java may feel familiar with the above code, almost self-explanatory. Let’s explain roughly, according to the code corresponding to the label:

  1. The js VM will define an empty default constructor if it is not defined.
  2. Es6 does not use the “function” keyword. All functions defined in a class are placed in the prototype of the class, so the class itself is just a syntactic sugar.
  3. The constructor calls the superclass constructor through super(). If there is a super method, the first statement executed in the constructor is required. This keyword is available only after super is called.
  4. Static methods, which can only be called by the class name outside the class definition, can be called by this inside, and static functions are inherited. As in the example: sTest is a static function defined in Person that can be passedMan.sTest()Direct call.

Es6 and ES5 inheritance differences

In most browsers’ ES5 implementations, each object has a __proto__ attribute that points to the prototype attribute of the corresponding constructor. Class is the syntactic sugar of the constructor and has both the Prototype and __proto__ attributes, so there are two inheritance chains.

(1) The __proto__ attribute of a subclass, which indicates the constructor’s inheritance, always points to the parent class.

(2) The prototype attribute’s __proto__ attribute, indicating method inheritance, always points to the prototype attribute of the parent class.

class A {
}

class B extends A {
}

B.__proto__ === A // true
B.prototype.__proto__ === A.prototype // true

Copy the code

In the code above, the __proto__ attribute of subclass B points to parent A, and the __proto__ attribute of subclass B’s Prototype attribute points to the prototype attribute of parent A.

The result is that class inheritance is implemented according to the following pattern:

Class A {} class B {} // an instance of B inherits an instance of object.setProtoTypeof (b.prototype, a.prototype); Object.setprototypeof (B, A);Copy the code

A simple implementation of Object.setPrototypeof is as follows:

Object.setPrototypeOf = function (obj, proto) {
  obj.__proto__ = proto;
  return obj;
}

Copy the code

Therefore, we get the result above.

Object.setPrototypeOf(B.prototype, A.prototype); // equivalent to b.prototype.__proto__ = a.prototype; Object.setPrototypeOf(B, A); // equivalent to b. __proto__ = A;Copy the code

The two inheritance chains can be understood as follows: as an object, the prototype (__proto__ attribute) of subclass (B) is the parent (A); As a constructor, the prototype (prototype property) of subclass (B) is an instance of the parent class.

Object.create(A.prototype); // equivalent to b.prototype.__proto__ = a.prototype;Copy the code

Deficiencies of ES6 inheritance

  1. Static attributes (except functions) are not supported.
  2. Private variables and functions cannot be defined in class. All functions defined in class will be put back into the stereotype, will be inherited by subclasses, and properties will be attached to this as instance properties. If a subclass wants to define a private method or a private variable, it cannot do so directly inside class braces. This is really inconvenient!

In summary, COMPARED to ES5, ES6 provides some object-oriented support at the language level, which is mostly just a syntactic sugar, but is easier to use, more semantic, more intuitive, and provides a standard way for javascript inheritance. Another important point is that ES6 supports native object inheritance.

polymorphism

A Polymorphism literally means “many states.” In object-oriented languages, interfaces can be implemented in many different ways, called polymorphisms. This is a standard definition, in c + + implementation of polymorphic way has a virtual function, abstract class, template, more violent in Java, all functions are “empty”, a subclass can override, in Java without the concept of virtual functions, of course, we are the same signature, parents and children can have different implementation of function call virtual functions, Virtual functions and templates (generics in Java) are the main ways to support polymorphism. Since there are no templates in javascript, we will only talk about virtual functions. Here is an example:

function Person(name,age){ this.name=name this.age=age } Person.prototype.toString=function(){ return "I am a Person, my name is "+ this.name } function Man(name,age){ Person.apply(this,arguments) } Man.prototype = Object.create(Person.prototype); Man.prototype.toString=function(){ return "I am a Man, my name is"+this.name; } var person=new Person("Neo",19) var man1=new Man("Davin",18) var man2=new Man("Jack",19) > person+"" > "I am a Person, My name is Neo" > man1+"" > "I am a Man, my name isDavin" > man1<man2Copy the code

In the example above, we respectively in subclasses and the parent class implements the toString method, in fact, the above code in js principle is very simple, for the same function, a subclass of the parent class, this feature is virtual functions, but does not distinguish between parameters in js number, also does not distinguish parameter type, only the function name, if the same name will be covered. Now let’s look at comment 1. We want to use the comparison operator directly to compare the size (by age) of two MEN. Operator overloading is available in c++, but not in Java or js. Fortunately, js can be implemented in a workaround way:

function Person(name,age){ this.name=name this.age=age } Person.prototype.valueOf=function(){ return this.age } function  Man(name,age){ Person.apply(this,arguments) } Man.prototype = Object.create(Person.prototype); var person=new Person("Neo",19) var man1=new Man("Davin",18) var man2=new Man("Jack",19) var man3=new Man("Joe",19) >man1<19//1 >true >person==19//2 >true >man1<man2//3 >true >man2==man3 //4 Note >true >person==man2//5 >falseCopy the code

Where 1, 2, 3 and 5 are determined under all JS VMS. But four is not necessarily! Javascript states that for comparison operators, if one value is an object and the other is a number, valueOf will be tried first, and toString will be called if valueOf is not specified. If it is a string, toString is tried first, if not, valueOf is tried, and if neither is specified, a type error exception is thrown. If both values are objects, then the reference address of the object is being compared, so if the object is itself = = = itself, all other cases are false. Now let’s go back to the sample code. The first three are standard behaviors. The fourth one depends on the implementation of the browser. If strictly according to the standard, this should be a Chrome bug. However, our code uses the double equal sign, which is not strict equality judgment, so the browser equality rule will be relaxed. It is worth mentioning that the result is false even though the person and man2 ages are both 19. To summarize, Chrome’s comparison strategy for instances of the same class is to try to convert and then compare the size, while for instances that are not of the same class, it returns false without doing any conversion. So my suggestion is: if the number and class instance comparison, is always safe, can rest assured to play, if it is between the same instance, can carry out unequal comparison, this result can be guaranteed, do not carry out equal comparison, the result is not guaranteed, general equal comparison, flexible approach is:

var equal= ! (ob1 < ob2 | | ob1 > ob2) / / no less than is not greater than, is equal to the premise is the object on both sides of the comparison operators to implement the valueOf or toStringCopy the code

Of course, there is a toJson method similar to toString and valueOf, but it has nothing to do with overloading, so it is not redundant.

Mathematical operator

Making objects support mathematical operators is essentially the same as making objects support comparison operators, with valueOf and toString conversions at the bottom. But there is a big limitation to the operator overloading simulated by overriding the original method: return values can only be numbers! The result of operator overloading in c++ can be an object. If we were to implement an addition to a complex class, which includes real and imaginary parts, and the addition was applied to both parts, the result of the addition (the return value) would still be a complex object, javascript wouldn’t be able to do anything about it.