[!NOTE]
能熟練掌握每種繼承方式的手寫實現,並知道該繼承實現方式的優缺點。javascript
function Parent() { this.name = 'zhangsan'; this.children = ['A', 'B', 'C']; } Parent.prototype.getName = function() { console.log(this.name); } function Child() { } Child.prototype = new Parent(); var child = new Child(); console.log(child.getName());
[!NOTE]
主要問題:
1. 引用類型的屬性被全部實例共享(this.children.push('name'))
2. 在建立Child的實例的時候,不能向Parent傳參java
function Parent(age) { this.names = ['zhangsan', 'lisi']; this.age = age; this.getName = function() { return this.names; } this.getAge = function() { return this.age; } } function Child(age) { Parent.call(this, age); } var child = new Child(18); child.names.push('haha'); console.log(child.names); var child2 = new Child(20); child2.names.push('yaya'); console.log(child2.names);
[!NOTE]
優勢:
1. 避免了引用類型的屬性被全部實例共享
2. 能夠直接在Child中向Parent傳參
缺點:
方法都在構造函數中定義了,每次建立實例都會建立一遍方法數組
/** * 父類構造函數 * @param name * @constructor */ function Parent(name) { this.name = name; this.colors = ['red', 'green', 'blue']; } Parent.prototype.getName = function() { console.log(this.name); } // child function Child(name, age) { Parent.call(this, name); this.age = age; } Child.prototype = new Parent(); // 校訂child的構造函數 Child.prototype.constructor = Child; // 建立實例 var child1 = new Child('zhangsan', 18); child1.colors.push('orange'); console.log(child1.name, child1.age, child1.colors); // zhangsan 18 (4) ["red", "green", "blue", "orange"] var child2 = new Child('lisi', 28); console.log(child2.name, child2.age, child2.colors); // lisi 28 (3) ["red", "green", "blue"]
[!NOTE]
優勢: 融合了原型鏈繼承和構造函數的優勢,是Javascript中最經常使用的繼承模式安全
------ 高級繼承的實現app
function createObj(o) { function F(){}; // 關鍵:將傳入的對象做爲建立對象的原型 F.prototype = o; return new F(); } // test var person = { name: 'zhangsan', friends: ['lisi', 'wangwu'] } var person1 = createObj(person); var person2 = createObj(person); person1.name = 'wangdachui'; console.log(person1.name, person2.name); // wangdachui, zhangsan person1.friends.push('songxiaobao'); console.log(person2.friends); // lisi wangwu songxiaobao
[!WARNING]
缺點:
對於引用類型的屬性值始終都會共享相應的值,和原型鏈繼承同樣函數
// 建立一個用於封裝繼承過程的函數,這個函數在內部以某種形式來加強對象 function createObj(o) { var clone = Object.create(o); clone.sayName = function() { console.log('say HelloWorld'); } return clone; }
[!WARNING]
缺點:與借用構造函數模式同樣,每次建立對象都會建立一遍方法優化
function Parent(name) { this.name = name; this.colors = ['red', 'green', 'blue']; } Parent.prototype.getName = function() { console.log(this, name); } function Child(name, age) { Parent.call(this, name); this.age = age; } // test1: // 1. 設置子類實例的時候會調用父類的構造函數 Child.prototype = new Parent(); // 2. 建立子類實例的時候也會調用父類的構造函數 var child1 = new Child('zhangsan', 18); // Parent.call(this, name); // 思考:如何減小父類構造函數的調用次數呢? var F = function(){}; F.prototype = Parent.prototype; Child.prototype = new F(); // 思考:下面的這一句話能夠嗎? /* 分析:由於此時Child.prototype和Parent.prototype此時指向的是同一個對象, 所以部分數據至關於此時是共享的(引用)。 好比此時增長 Child.prototype.testProp = 1; 同時會影響 Parent.prototype 的屬性的。 若是不模擬,直接上 es5 的話應該是下面這樣吧 Child.prototype = Object.create(Parent.prototype);*/ Child.prototype = Parent.prototype; // 上面的三句話能夠簡化爲下面的一句話 Child.prototype = Object.create(Parent.prototype); // test2: var child2 = new Child('lisi', 24);
// 自封裝一個繼承的方法 function object(o) { // 下面的三句話實際上就是相似於:var o = Object.create(o.prototype) function F(){}; F.prototype = o.prototype; return new F(); } function prototype(child, parent) { var prototype = object(parent.prototype); // 維護原型對象prototype裏面的constructor屬性 prototype.constructor = child; child.prototype = prototype; } // 調用的時候 prototype(Child, Parent)
var o1 = {name: 'value'}; var o2 = new Object({name: 'value'}); var M = function() {this.name = 'o3'}; var o3 = new M(); var P = {name: 'o4'}; var o4 = Object.create(P)
__proto__
內部屬性,這個屬性所對應的就是該對象的原型__proto__
以外,還預置了 prototype 屬性__proto__
。任何一個實例對象經過原型鏈能夠找到它對應的原型對象,原型對象上面!ui
的實例和方法都是實例所共享的。this
一個對象在查找以一個方法或屬性時,他會先在本身的對象上去找,找不到時,他會沿着原型鏈依次向上查找。es5
注意: 函數纔有prototype,實例對象只有有__proto__, 而函數有的__proto__是由於函數是Function的實例對象
判斷實例對象的__proto__屬性與構造函數的prototype是否是用一個引用。若是不是,他會沿着對象的__proto__向上查找的,直到頂端Object。
使用對象.construcor
直接可判斷
var obj = {}; obj.__proto__ = Base.prototype; Base.call(obj);
類的聲明
// 普通寫法 function Animal() { this.name = 'name' } // ES6 class Animal2 { constructor () { this.name = 'name'; } }
在構造函數中 使用Parent.call(this)
的方法繼承父類屬性。
原理: 將子類的this使用父類的構造函數跑一遍
缺點: Parent原型鏈上的屬性和方法並不會被子類繼承
function Parent() { this.name = 'parent' } function Child() { Parent.call(this); this.type = 'child' }
原理:把子類的prototype(原型對象)直接設置爲父類的實例
缺點:由於子類只進行一次原型更改,因此子類的全部實例保存的是同一個父類的值。
當子類對象上進行值修改時,若是是修改的原始類型的值,那麼會在實例上新建這樣一個值;
但若是是引用類型的話,他就會去修改子類上惟一一個父類實例裏面的這個引用類型,這會影響全部子類實例
function Parent() { this.name = 'parent' this.arr = [1,2,3] } function Child() { this.type = 'child' } Child.prototype = new Parent(); var c1 = new Child(); var c2 = new Child(); c1.__proto__ === c2.__proto__
組合構造函數中使用call繼承和原型鏈繼承。
原理: 子類構造函數中使用Parent.call(this);
的方式能夠繼承寫在父類構造函數中this上綁定的各屬性和方法;
使用Child.prototype = new Parent()
的方式能夠繼承掛在在父類原型上的各屬性和方法
缺點: 父類構造函數在子類構造函數中執行了一次,在子類綁定原型時又執行了一次
function Parent() { this.name = 'parent' this.arr = [1,2,3] } function Child() { Parent.call(this); this.type = 'child' } Child.prototype = new Parent();
由於這時父類構造函數的方法已經被執行過了,只須要關心原型鏈上的屬性和方法了
Child.prototype = Parent.prototype;
缺點:
constructor
,此時直接使用父類的prototype的話那麼會致使 實例的constructor爲Parent,即不能區分這個實例對象是Child的實例仍是父類的實例對象。注意:這個時候instanseof是能夠判斷出實例爲Child的實例的,由於instanceof的原理是沿着對象的__proto__判斷是否有一個原型是等於該構造函數的原型的。這裏把Child的原型直接設置爲了父類的原型,那麼: 實例.__proto__ === Child.prototype === Child.prototype
function Parent() { this.name = 'parent' this.arr = [1,2,3] } function Child() { Parent.call(this); this.type = 'child' } Child.prototype = Object.create(Parent.prototype); //提供__proto__ Child.prototype.constrctor = Child;
Object.create()方法建立一個新對象,使用現有的對象來提供新建立的對象的__proto__
/** * 工廠模式建立對象 * @param name * @return {Object} */ function createPerson(name){ var o = new Object(); o.name = name; o.getName = function() { console.log(this.name); } return o; } var person = createPerson('zhangsan'); console.log(person.__proto__ === Object.prototype); // true
缺點:沒法識別當前的對象,由於建立的全部對象實例都指向的是同一個原型
/** * 使用構造函數的方式來建立對象 * @param name * @constructor */ function Person(name) { this.name = name; this.getName = function() { console.log(this.name) } } var person = new Person('lisi'); console.log(person.__proto__ === Person.prototype)
優勢:實例剋識別僞一個特定的類型
缺點:每次建立實例對象的時候,每一個方法都會被建立一次
function Person(name) { this.name = name; this.getName = getName; } function getName() { console.log(this.name); } var person = new Person('zhangsan'); console.log(person.__proto__ === Person.prototype);
優勢:解決了每一個方法都要被從新建立的問題
缺點:不合乎代碼規範……
function Person(name) { } Person.prototype.name = 'lisi'; Person.prototype.getName = function() { console.log(this.name); } var person = new Person(); console.log(Person.prototype.constructor) // Person
優勢:方法不會被從新建立
缺點:1. 全部的屬性和方法全部的實例上面都是共享的;2. 不能初始化參數
function Person(name) { } Person.prototype = { name: 'lisi', getName: function() { console.log(this.name); } } var person = new Person(); console.log(Person.prototype.constructor) // Object console.log(person.constructor == person.__proto__.constructor) // true
優勢:封裝性好了一些
缺點:重寫了Person的原型prototype屬性,丟失了原始的prototype上的constructor屬性
function Person(name) { } Person.prototype = { constructor: Person, name: 'lisi', getName: function() { console.log(this.name) } } var person = new Person();
優勢:實例能夠經過constructor屬性找到所屬的構造函數
缺點:全部的屬性和方法都共享,並且不能初始化參數
function Person(name) { this.name = name; } Person.prototype = { constructor: Person, getName: function() { console.log(this.name) } } var person = new Person('zhangsan');
優勢:基本符合預期,屬性私有,方法共享,是目前使用最普遍的方式
缺點:方法和屬性沒有寫在一塊兒,封裝性不是太好
// 第一種建立思路: function Person(name) { this.name = name; if (typeof this.getName !== 'function') { Person.prototype.getName = function() { console.log(this.name); } } } var person = new Person(); // 第二種建立的思路:使用對象字面量重寫原型上的方法 function Person(name) { this.name = name; if (typeof this.getName !== 'function') { Person.prototype = { constructor: Person, getName: function() { console.log(this.name) } } return new Person(name); } } var person1 = new Person('zhangsan'); var person2 = new Person('lisi'); console.log(person1.getName()); console.log(person2.getName());
/** * 寄生構造函數模式 * @param name * @return {Object} * @constructor */ function Person(name){ var o = new Object(); o.name = name; o.getName = function() { console.log(this.name) } return o; } var person = new Person('zhangsan'); console.log(person instanceof Person); // false console.log(person instanceof Object); // true // 使用寄生-構造函數-模式來建立一個自定義的數組 /** * 特殊數組的構造器 * @constructor */ function SpecialArray() { var values = new Array(); /*for (var i = 0, len = arguments.length; i < len; i++) { values.push(arguments[i]); }*/ // 開始添加數據(能夠直接使用apply的方式來優化代碼) values.push.apply(values, arguments); // 新增的方法 values.toPipedString = function(){ return this.join('|'); } return values; } // 使用new來建立對象 var colors1 = new SpecialArray('red1', 'green1', 'blue1'); // 不使用new來建立對象 var colors2 = SpecialArray('red2', 'green2', 'blue2'); console.log(colors1, colors1.toPipedString()); console.log(colors2, colors2.toPipedString());
/** * 穩妥的建立對象的方式 * @param name * @return {number} * @constructor */ function Person(name){ var o = new Object(); o.sayName = function() { // 這裏有點相似於在一個函數裏面使用外部的變量 // 這裏直接輸出的是name console.log(name); } return o; } var person = Person('lisi'); person.sayName(); person.name = 'zhangsan'; person.sayName(); console.log(person instanceof Person); // false console.log(person instanceof Object); // false
[!NOTE] 與寄生的模式的不一樣點:1. 新建立的實例方法不引用this 2.不使用new操做符調用構造函數 優勢:最適合一些安全的環境中使用 缺點:和工廠模式同樣,是沒法識別對象的所屬類型的