JavaScript的7种继承模式
《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属性为引用类型,可能存在子对象意外覆盖父对象属性的风险。
// 演示缺点1
var 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; // Partick
typeof 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; // Partick
kid.say(); // Partick
delete 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(); // undefined
kid.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); // string
console.log(typeof kid.getName); // function
// 继承方式2:仅继承父原型的属性
var kid = object(Parent.prototype);
console.log(typeof kid.name); // undefined
console.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,4
dad.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,3
dad.reads === kid.reads; // false
混入
从多个对象中复制出任意成员,并将这些成员组合成一个新的对象。
遇到同名属性,总是使用靠后对象的值,即越往后优先级越高。
function mix() {
var i, prop, child = {};
for (i = 0; i<arguments.length; i++) {
for (prop in arguments[i]) {
if (arguments[i].hasOwnProperty(prop)) {
child[prop] = arguments[i][prop];
}
}
}
return child;
}
// 测试
var cake = mix(
{ eggs: 2, large: true },
{ buter: 1, saleted: true },
{ flour: "3 cups" },
{ sugar: "sure!" },
{ eggs: 3 } // 同名属性,越往后优先级越高
);
console.dir(cake);
JavaScript的7种继承模式的更多相关文章
- JavaScript的3种继承方式
JavaScript的继承方式有多种,这里列举3种,分别是原型继承.类继承以及混合继承. 1.原型继承 优点:既继承了父类的模板,又继承了父类的原型对象: 缺点:不是子类实例传参,而是需要通过父类实例 ...
- JavaScript 对象的创建和对6种继承模式的理解和遐想
JS中总共有六种继承模式,包括原型链.借用构造函数.组合继承.原型式继承寄生式继承和寄生组合式继承.为了便于理解记忆,我遐想了一个过程,对6中模式进行了简单的阐述. 很长的一个故事,姑且起个名字叫 ...
- JavaScript常用八种继承方案
更新:在常用七种继承方案的基础之上增加了ES6的类继承,所以现在变成八种啦,欢迎加高级前端进阶群一起学习(文末). --- 2018.10.30 1.原型链继承 构造函数.原型和实例之间的关系:每个构 ...
- JavaScript的几种继承方式
看<JavaScript高级程序设计>做的一些笔记 ECMAScript只支持实现继承,不支持接口继承(因为函数没有签名) 原型链(实现继承的主要方法): function SuperTy ...
- Javascript的四种继承方式
在Javascript中,所有开发者定义的类都可以作为基类,但出于安全性考虑,本地类和宿主类不能作为基类,这样可以防止公用访问编译过的浏览器级的代码,因为这些代码可以被用于恶意攻击. 选定基类后,就可 ...
- 都0202年了,你还不知道javascript有几种继承方式?
前言 当面试官问你:你了解js哪些继承方式?es6的class继承是如何实现的?你心中有很清晰的答案吗?如果没有的话,可以通过阅读本文,帮助你更深刻地理解js的所有继承方式. js ...
- javascript的几种继承
1.原型链继承:构造函数.原型和实例的关系:每个构造函数都有一个原型对象,原型对象都包含一个指向构造函数的指针,而实例都包含一个指向原型对象的内部指针.确认原型和实例之间的关系用instanceof. ...
- 浅谈 JavaScript 中的继承模式
最近在读一本设计模式的书,书中的开头部分就讲了一下 JavaScript 中的继承,阅读之后写下了这篇博客作为笔记.毕竟好记性不如烂笔头. JavaScript 是一门面向对象的语言,但是 ES6 之 ...
- JavaScript几种继承方式的总结
1.原型链继承 直接将子类型的原型指向父类型的实例,即"子类型.prototype = new 父类型();",实现方法如下: //父类构造函数 function father(n ...
随机推荐
- Vue折腾记 - (3)写一个不大靠谱的typeahead组件
Vue折腾记 - (3)写一个不大靠谱的typeahead组件 2017年07月20日 15:17:05 阅读数:691 前言 typeahead在网站中的应用很多..今天跟着我来写一个不大靠谱的ty ...
- 【转】webpack4
1.不再支持node.js4.X 2.不能用webpack命令直接打包指定的文件,只能使用webpack.config.js进行配置. 即:webpack demo01.js bundle01.j ...
- DPDK如何抓包
原创翻译,转载请注明出处. DPDK的librte_pdump库,提供了在DPDK框架下抓包的功能.这个库通过完全复制Rx和Tx的mbuf到一个新的内存池,因此它降低应用程序的性能,所以只推荐在调试的 ...
- Android调用Java WebSevice篇之一
一.服务端WebService 1.服务端环境配置 MyEclipse 10.0.Tomcat6.0.JDK6.0. 2.下载axis相关jar包. 3.创建webservice. ...
- 算法(12)Pascal's Triangle II
题目:输出帕斯卡三角的第k行 思路:真没思路,发现几个easy的题不容易想!这里的大致思路是从后开始更新第k行!
- [剑指Offer] 17.树的子结构
题目描述 输入两棵二叉树A,B,判断B是不是A的子结构.(ps:我们约定空树不是任意一个树的子结构) [思路]要查找树A中是否存在和树B结构一样的子树,可以分成两步: 1.第一步在树A中找到和B的根节 ...
- hdu 3648 Median Filter (树状数组)
Median Filter Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Tot ...
- P2874 [USACO07FEB]新牛棚Building A New Barn
题目描述 After scrimping and saving for years, Farmer John has decided to build a new barn. He wants the ...
- 【南开OJ2264】节操大师(贪心+二分+并查集/平衡树)
好久没更新了,今天就随便写一个吧 题目内容 MK和他的小伙伴们(共n人,且保证n为2的正整数幂)想要比试一下谁更有节操,于是他们组织了一场节操淘汰赛.他们的比赛规则简单而暴力:两人的节操正面相撞,碎的 ...
- 什么是Redis的事务
一.什么是Redis的事务 可以一次执行多个命令,本质上是一组命令的集合.一个事务中的所有命令都会序列化,然后按顺序地串行化执行,而不会被插入其它命令. 二.Redis的事务可以做什么 一个队列中,一 ...