《JavaScript模式》一书中,对于JavaScript的几种继承模式讲解得很清楚,给我提供了很大帮助。总结一下,有如下7种模式。
继承模式1——设置原型(默认模式)
实现方式:
// 父构造函数function Parent(name) { this.name = name || 'Adam';}// 向原型中添加方法Parent.prototype.say = function() { return this.name;}// 子构造函数(空白)function Child(name) {}// 继承:设置原型Child.prototype = new Parent();// 测试var kid = new Child();kid.say(); // Adam
原型链:
注意:
__proto__
属性仅用来解释原型链,不可用于开发中。- 若子对象#3定义属性name,并不会修改父对象#2的name属性,而是直接在子对象#3上创建一个自身属性。如果使用delete删除子对象#3的name属性,那么父对象#2的name属性将表现出来。
- 优点:
- 子对象继承了:父构造函数中的this属性、父原型中的属性。
- 缺点:
- 不支持将参数传递到子构造函数中,而子构造函数然后又将参数传递到父构造函数中。
- 如果父类构造函数中的this属性为引用类型,可能存在子对象意外覆盖父对象属性的风险。
// 演示缺点1var s = new Child('Seth');s.say(); // Adam
// 演示缺点2// 父构造函数function Article() { this.tags = ['js', 'css'];}var article = new Article();// 子构造函数及继承function Blog() {}Blog.prototype = article;// 子对象意外修改父对象的引用属性var blog = new Blog();blog.tags.push('html');console.log(article.tags.join(' ')); // js css html
继承模式2——借用构造函数
实现方式:
// 父构造函数function Parent(name) { this.name = name || 'Adam';}// 向原型中添加方法Parent.prototype.say = function() { return this.name;}// 子构造函数function Child(name) { // 继承:借用构造函数 Parent.apply(this, arguments);}// 测试var kid = new Child('Partrick');kid.name; // Particktypeof kid.say; // undefined
原型链:
注意:
- 缺点:只能继承父构造函数中的this属性,不能继承父原型中的属性。
- 优点:
- 本模式解决了从子构造函数到父构造函数的参数传递问题。
- 子对象可以获得父对象自身成员的副本(而非引用),并且不会存在子对象意外覆盖父对象属性的风险。
继承模式3——设置原型&借用构造函数
实现方式:
// 父构造函数function Parent(name) { this.name = name || 'Adam';}// 向原型中添加方法Parent.prototype.say = function() { return this.name;}// 子构造函数function Child(name) { // 继承:借用构造函数 Parent.apply(this, arguments);}// 继承:设置原型Child.prototype = new Parent();// 测试var kid = new Child('Partrick');kid.name; // Partickkid.say(); // Partickdelete kid.name;kid.say(); // Adam
原型链:
注意:
- 优点:
- 能够获得父对象自身成员的副本。子对象可以安全地修改自身属性,且不会带来修改其父对象的风险。
- 子对象继承了:父构造函数中的this属性、父原型中的属性。
- 子构造函数可将任意参数传递到父构造函数中。
- 缺点:
- 父构造函数被调用了两次,导致其效率低下。自身的属性(name)被继承了两次,删除了子类本身的name属性的副本后,原型链上的name属性将表现出来。
继承模式4——共享原型
实现方式:
// 父构造函数function Parent(name) { this.name = name || 'Adam';}// 向原型中添加方法Parent.prototype.say = function() { return this.name;}// 子构造函数function Child() {}// 继承:共享原型child.prototype = Parent.prototype;
原型链:
注意:
- 本模式适用于:可复用成员应转移到原型中,而不是放置在父类this中。任何值得继承的东西都应该放置在原型中实现。
- 不能继承父构造函数中的this属性,只能继承父原型中的属性。
- 缺点:
- 如果在继承链下方的某处存在一个子对象或孙子对象修改了原型,将会影响到所有父对象和祖先对象。
继承模式5——临时构造函数
实现方式:
// 父构造函数function Parent(name) { this.name = name || 'Adam';}// 向原型中添加方法Parent.prototype.say = function() { return this.name;}// 子构造函数function Child(name) {}// 继承:设置原型inherit(Child, Parent);// 实现:function inherit(C, P) { var F = function() {}; F.prototype = P.prototype; C.prototype = new F(); C.prototype.constructor = C;}// 优化:避免在每次需要继承时,都创建临时(代理)构造函数。// 实现:即时函数+闭包var inherit2 = (function() { var F = function() {}; return function(C, P) { F.prototype = P.prototype; C.prototype = new F(); C.prototype.constructor = C; }})();// 测试var kid = new Child();kid.say(); // undefinedkid.name = "Peter";kid.say(); // Peter
原型链:
注意:
- 子对象仅继承了原型的属性。原型仅用来放置可复用的功能。父构造函数的this中的任何成员都不会被继承。
- 需要重置子构造函数的指针:
C.prototype.constructor = C
继承模式6——原型继承
实现方式:
function object(P) { var F = function() {}; F.prototype = P; return new F();}
对象字面量方式创建父对象
var parent = { name: "papa"}var child = object(parent);// 测试console.log(child.name);
构造函数方式创建父对象
// 父构造函数function Parent() { this.name = "papa";}Parent.prototype.getName = function() { return this.name;}// 创建一个父对象var papa = new Parent();// 继承方式1:父构造函数中的this属性、父原型的属性都被继承var kid = object(papa);console.log(typeof kid.name); // stringconsole.log(typeof kid.getName); // function// 继承方式2:仅继承父原型的属性var kid = object(Parent.prototype);console.log(typeof kid.name); // undefinedconsole.log(typeof kid.getName); // function
ES5: Object.create()
继承模式7——复制属性
浅复制
在使用浅复制时,如果改变了子对象的属性,并且该属性恰好是一个对象,那么这种操作也将修改父对象。
function extend(parent, child) { var i; child = child || {}; for (i in parent) { if (parent.hasOwnProperty(i)) { child[i] = parent[i]; } } return child;}// 测试var dad = { counts: [1, 2, 3], reads: { paper: true }};var kid = extend(dad);kid.counts.push(4);dad.counts.toString(); // 1,2,3,4dad.reads === kid.reads; // true
深复制
检查父对象的某个属性是否为对象,如果是,则需要递归复制出该对象的属性。
function extendDeep(parent, child) { var i, toStr = Object.prototype.toString, astr = "[object Array]"; child = child || {}; for (i in parent) { if (parent.hasOwnProperty(i)) { if (typeof parent[i] === 'object') { child[i] = (toStr.call(parent[i]) === astr) ? [] : {}; extendDeep(parent[i], child[i]); } else { child[i] = parent[i]; } } } return child;}// 测试var dad = { counts: [1, 2, 3], reads: { paper: true }};var kid = extendDeep(dad);kid.counts.push(4);dad.counts.toString(); // 1,2,3dad.reads === kid.reads; // false
混入
从多个对象中复制出任意成员,并将这些成员组合成一个新的对象。
遇到同名属性,总是使用靠后对象的值,即越往后优先级越高。function mix() { var i, prop, child = {}; for (i = 0; i