重温Javascript继承机制(3)
作者:瓜瓜 发布时间:2011-07-04 12:17:23
三、构造函数如何实现继承
现在有一个”MED”对象的构造函数(MED:Marketing Experience Design,营销体验设计)
function MED(){
this.aim = "营销体验设计";
}
依然是”WD”对象的构造函数,
function WD(skill,sex){
this.skill = skill;
this.sex = sex;
}
怎样才能使”WD”继承”MED”呢?
1. apply绑定构造函数实现
最简单的方法,大概就是使用call或apply方法,将父对象的构造函数绑定在子对象上,也就是在子对象构造函数中加一行:
function WD(skill,sex){
MED.apply(this, arguments);
this.skill = skill;
this.sex = sex;
}
var WD1 = new WD("Html","男");
console.log(WD1.aim); // "营销体验设计"
2. prototype模式实现
我们通常的做法是使用prototype属性。如果”WD”的prototype对象,指向一个MED的实例,那么所有”WD”的实例,就能继承MED了。
WD.prototype = new MED();//我们将WD的prototype对象指向一个MED的实例。
WD.prototype.constructor = WD;
var WD1 = new WD("Html","男");
console.log(WD1.aim); // 营销体验设计
这句
WD.prototype = new MED();
相当于完全删除了prototype 对象原先的值,然后赋予一个新值。那么第二行又是什么意思呢?
WD.prototype.constructor = WD;
原来,任何一个prototype对象都有一个constructor属性,指向它的构造函数。也就是说,WD.prototype 这个对象的constructor属性,是指向WD的。
我们在前一步已经删除了这个prototype对象原来的值,所以新的prototype对象没有constructor属性,需要我们手动加上去,否则后面的”继承链”会出问题。这就是第二行的意思。
注意,这是很重要的一点,编程时务必要遵守,下文都遵循这一点,即如果替换了prototype对象,
o.prototype = {};
那么,下一步必然是为新的prototype对象加上constructor属性,并将这个属性指回原来的构造函数。
o.prototype.constructor = o;
3. 从prototype直接继承实现
由于MED对象中,不变的属性都可以直接写入MED.prototype。所以,我们也可以让WD()跳过 MED(),直接继承MED.prototype。
现在,我们先将MED对象改写:
function MED(){ }
MED.prototype.skill = "MED";
然后,将WD的prototype对象指向MED的prototype对象,这样就完成了继承。
WD.prototype = MED.prototype;
WD.prototype.constructor = WD;
var WD1 = new WD("Html","男");
console.log(WD1.skill); // MED
与前一种方法相比,这样做的优点是效率比较高(不用执行和建立MED的实例了),比较省内存。缺点是 WD.prototype和MED.prototype现在指向了同一个对象,那么任何对WD.prototype的修改,都会反映到MED.prototype。
所以,上面这一段代码其实是有问题的。请看第二行
WD.prototype.constructor = WD;
这一句实际上把MED.prototype对象的constructor属性也改掉了!
console.log(MED.prototype.constructor); // WD
4. 利用一个空对象作为中介来实现
由于”直接继承prototype”存在上述的缺点,所以可以利用一个空对象作为中介。
var F = function(){};
F.prototype = MED.prototype;
WD.prototype = new F();
WD.prototype.constructor = WD;
F是空对象,所以几乎不占内存。这时,修改WD的prototype对象,就不会影响到MED的prototype对象。
console.log(MED.prototype.constructor); // MED
5.利用 prototype模式的封装函数
我们将上面的方法,封装成一个函数,便于使用。
function extend(Child, Parent) {
var F = function(){};
F.prototype = Parent.prototype;
Child.prototype = new F();
Child.prototype.constructor = Child;
}
使用的时候,方法如下
extend(WD,MED);
var WD1 = new WD("Html","男");
console.log(WD1.aim); // 营销体验设计
这个extend函数就是YUI库如何实现继承的方法。
6. 拷贝继承实现
上面是采用prototype方式来实现继承。其实既然子对象会拥有父对象的属性和方法,我们直接采用”拷贝”方法也可以达到效果。简单说,如果把父对象的所有属性和方法,拷贝进子对象,不也能够实现继承吗?
首先,还是把MED的所有不变属性,都放到它的prototype对象上。
function MED(){}
MED.prototype.aim = "营销体验设计";
然后,再写一个函数,实现属性拷贝的目的。
function extendCopy(Child, Parent) {
var p = Parent.prototype;
var c = Child.prototype;
for (var i in p) {
c[i] = p[i];
}
}
这个函数的作用,就是将父对象的prototype对象中的属性,一一拷贝给Child对象的prototype对象。
使用的时候,这样写:
extendCopy(WD, MED);
var WD1 = new WD("Html","男");
console.log(WD1.aim); // 营销体验设计
猜你喜欢
- 客户端调用XMLHTTP的过程很简单,只有5个步骤: 1、创建XMLHTTP对象 2、打开与服务端的连接,同时定义指令发送方式,服务网页(U
- 在不久前的一天,当我为了解决一个语法问题来翻阅VBscript文档时,偶然间发现在了下面的一句话: &nb
- 很实用的过滤重复数据的asp代码,函数如下:<%'**************************************
- 这是个老话题了,之所以再拿出来说,是因为浏览器一直在进步,以前最好的方法现在不一定是最好的。1 如何进行字符串连接?首先让我们来回顾一下字符
- 当然这应该属于正常过滤手法,而还有一种过滤HTML标签的最终极手法,则是将一对尖括号及尖括号中的所有字符均替换不显示,该方法对于内容中必须描
- 为什么很多站长开始做英文网站,我想主要是原因是良好的互联网环境让大家更容易赚到钱,中小站长做英文网站大致为两类,一是电子商务的外贸网站,二是
- [sql] -- ===================【创建存储过程】===================== USE [Message
- 最近心血来潮加上有点闲情,动手写了第一个JavaScript版的俄罗斯方块Easy Tetris.先上Easy Tetris俄罗斯方块游戏截
- 第一题:ASP中,VBScript的唯一的数据类型是什么?第二题:在ASP中,VBScript有多种控制程序流程语句,如If…Then, S
- 此处收集的是一些夺人心魄的创意广告牌,巧妙的构思十分值得大家观瞻.......超人归来 superman returnsNespressos
- 用mysqlbinlog.exe查看二进制日志是否启用了日志mysql>show variables like 'log%
- 在讲样式表开发管理之前,我想插播一个小知识。前几天看web标准设计组里,看到龍佑康同学问到关于 block 和 inline 的区别。记得以
- 1、使用mysqldump工具将MySql数据库备份mysqldump -u root -p -c --default-character-
- 1.5 学习ASP.net 的过程中如何求助--加入 ASPNG 讨论列表 Charles Carroll 作为不断壮大的 ASP.NET
- 今天刚看到ChinaUI.com新改版的网站,觉得优艾网是越来越漂亮了,这次的变化的确是挺大的。一早大概的看了一下就关了,后来晚上又打开看时
- 我们经常会遇到这样的问题你还在为你的MySQL命令模式下,前面的提示信息还是:mysql>,那么我们如何更改mysql命令下提示信息呢
- 关于本文:本着互联网共享主义精神,特写此文献给建站新手,授人以鱼,不如授人以渔,本文所讲的只是方法和原理,希望大家看完此文,能够从中得到些帮
- 代码如下:Function getCurrentUrl() On Error Resume Next Dim strTemp If LCas
- 问题:如何把具有相同字段的记录删除,只留下一条。 例如:表test里有id,name字段,如果有name相同的记录只留下一条,
- 你是否对获得MySQL改变字符集的实际操作感到十分头疼?不用急,以下的文章将会给你正确的解答方案,以下的文章主要是介绍获得MySQL改变字符