js的2种继承方式详解


Posted in Javascript onMarch 04, 2014

js中继承可以分为两种:对象冒充和原型链方式

一、对象冒充包括三种:临时属性方式、call()及apply()方式
1.临时属性方式

function Person(name){
     this.name = name;
     this.say = function(){
          alert('My name is '+this.name);
     }
}
function F2E(name,id){
     this.temp = Person;
     this.temp(name);
     delete this.temp;
     this.id = id;
     this.showId = function(){
          alert('Good morning,Sir,My work number is '+this.id);
     }
}
var simon = new F2E('Simon',9527);
simon.say();
simon.showId();

2.call()/apply()方式
实质上是改变了this指针的指向
function Person(name){
     this.name = name;
     this.say = function(){
          alert('My name is '+this.name);
     }
}
function F2E(name,id){
     Person.call(this,name); //apply()方式改成Person.apply(this,new Array(name));
     this.id = id;
     this.showId = function(){
          alert('Good morning,Sir,My work number is '+this.id);
     }
}
var simon = new F2E('Simon',9527);
simon.say();
simon.showId();

缺点:先来看这么一张内存分配图:

在OO概念中,new实例化后,对象就在堆内存中形成了自己的空间,值得注意的是,这个代码段。而成员方法就是存在这个代码段的,并且方法是共用的。问题就在这里,通过对象冒充方式继承时,所有的成员方法都是指向this的,也就是说new之后,每个实例将都会拥有这个成员方法,并不是共用的,这就造成了大量的内存浪费。并且通过对象冒充的方式,无法继承通过prototype方式定义的变量和方法,如以下代码将会出错:

function Person(name){
     this.name = name;
     this.say = function(){
          alert('My name is '+this.name);
     }
}
Person.prototype.age = 20;
Person.prototype.sayAge = function(){alert('My age is '+this.age)};function F2E(name,id){
     Person.apply(this,new Array(name));
     this.id = id;
     this.showId = function(){
          alert('Good morning,Sir,My work number is '+this.id);
     }
}
var simon = new F2E('Simon',9527);
simon.sayAge(); //提示TypeError: simon.sayAge is not a function

二、原型链方式
function Person(){
     this.name = 'Simon';
}
Person.prototype.say = function(){
     alert('My name is '+this.name);
}function F2E(id){
     this.id = id;
     this.showId = function(){
          alert('Good morning,Sir,My work number is '+this.id);
     }
}
F2E.prototype = new Person();
var simon = new F2E(9527);
simon.say();
simon.showId();
alert(simon.hasOwnProperty('id')); //检查是否为自身属性

接下来按照上面的例子来理解以下js原型链概念:

原型链可以理解成:js中每个对象均有一个隐藏的__proto__属性,一个实例化对象的__proto__属性指向其类的prototype方法,而这个prototype方法又可以被赋值成另一个实例化对象,这个对象的__proto__又需要指向其类,由此形成一条链,也就是前面代码中的

F2E.prototype = new Person()

这句是关键。js对象在读取某个属性时,会先查找自身属性,没有则再去依次查找原型链上对象的属性。也就是说原型链的方法是可以共用的,这样就解决了对象冒充浪费内存的缺点。

下面再来说缺点:
缺点显而易见,原型链方式继承,就是实例化子类时不能将参数传给父类,也就是为什么这个例子中function Person()没有参数,而是直接写成了this.name=”Simon”的原因。下面的代码将不能达到预期的效果:

function Person(name){
     this.name = name;
}
Person.prototype.say = function(){
     alert('My name is '+this.name);
}function F2E(name,id){
     this.id = id;
     this.showId = function(){
          alert('Good morning,Sir,My work number is '+this.id);
     }
}
F2E.prototype = new Person();
var simon = new F2E("Simon",9527);
simon.say();
simon.showId();
 
function Person(name){
     this.name = name;
}
Person.prototype.say = function(){
     alert('My name is '+this.name);
}
function F2E(name,id){
     this.id = id;
     this.showId = function(){
          alert('Good morning,Sir,My work number is '+this.id);
     }
}
F2E.prototype = new Person();  //此处无法进行传值,this.name或者name都不行,直接写F2E.prototype = new Person('wood')是可以的,但是这样的话simon.say()就变成了My name is wood
var simon = new F2E("Simon",9527);
simon.say();  //弹出 My name is undefined
simon.showId();

最后,总结一下自认为较好的继承实现方式,成员变量采用对象冒充方式,成员方法采用原型链方式,代码如下:

function Person(name){
     this.name = name;
}Person.prototype.say = function(){
     alert('My name is '+this.name);
}
function F2E(name,id){
     Person.call(this,name);
     this.id = id;
}
F2E.prototype = new Person(); 
//此处注意一个细节,showId不能写在F2E.prototype = new Person();前面
F2E.prototype.showId = function(){
     alert('Good morning,Sir,My work number is '+this.id);
}
var simon = new F2E("Simon",9527);
simon.say();
simon.showId();
Javascript 相关文章推荐
Mootools 1.2教程 定时器和哈希简介
Sep 15 Javascript
setinterval()与clearInterval()JS函数的调用方法
Jan 21 Javascript
Node.js实现数据推送
Apr 14 Javascript
基于js实现的限制文本框只可以输入数字
Dec 05 Javascript
jQuery加载及解析XML文件的方法实例分析
Jan 22 Javascript
AngularJS实现使用路由切换视图的方法
Jan 24 Javascript
解决iView中时间控件选择的时间总是少一天的问题
Mar 15 Javascript
详解写好JS条件语句的5条守则
Feb 28 Javascript
js常用正则表达式集锦
May 17 Javascript
vue input输入框关键字筛选检索列表数据展示
Oct 26 Javascript
JS中实现浅拷贝和深拷贝的代码详解
Jun 05 Javascript
解决vue项目本地启动时无法携带cookie的问题
Feb 06 Vue.js
Seajs的学习笔记
Mar 04 #Javascript
文本域中换行符的替换示例
Mar 04 #Javascript
为jquery的ajaxfileupload增加附加参数的方法
Mar 04 #Javascript
Document.location.href和.replace的区别示例介绍
Mar 04 #Javascript
LABjs、RequireJS、SeaJS的区别
Mar 04 #Javascript
js的Boolean对象初始值示例
Mar 04 #Javascript
js动态拼接正则表达式的两种方法
Mar 04 #Javascript
You might like
php数组查找函数in_array()、array_search()、array_key_exists()使用实例
2014/04/29 PHP
php实现mysql备份恢复分卷处理的方法
2014/12/26 PHP
thinkPHP实现多字段模糊匹配查询的方法
2016/12/01 PHP
详解thinkphp实现excel数据的导入导出(附完整案例)
2016/12/29 PHP
解析 thinkphp 框架中的部分方法
2017/05/07 PHP
PHP设计模式之观察者模式定义与用法示例
2018/08/04 PHP
PHP面向对象程序设计之接口的继承定义与用法详解
2018/12/20 PHP
一个小型js框架myJSFrame附API使用帮助
2008/06/28 Javascript
javascript 控制 html元素 显示/隐藏实现代码
2009/09/01 Javascript
如何正确使用javascript 来进行我们的程序开发
2014/06/23 Javascript
js添加select下默认的option的value和text的方法
2014/10/19 Javascript
上传文件返回的json数据会被提示下载问题解决方案
2014/12/03 Javascript
js实现图片和链接文字同步切换特效的方法
2015/02/20 Javascript
IE9+已经不对document.createElement向下兼容的解决方法
2015/09/14 Javascript
jquery实现模拟百分比进度条渐变效果代码
2015/10/29 Javascript
JS 在数组指定位置插入/删除数据的方法
2017/01/12 Javascript
node实现定时发送邮件的示例代码
2017/08/26 Javascript
JavaScript学习笔记之图片库案例分析
2019/01/08 Javascript
Vue组件通信的几种实现方法
2019/04/25 Javascript
深入浅析vue中cross-env的使用
2019/09/12 Javascript
[02:57]2014DOTA2国际邀请赛 选手辛苦解说更辛苦
2014/07/10 DOTA
[34:41]夜魇凡尔赛茶话会 第二期02:你画我猜
2021/03/11 DOTA
Windows下python2.7.8安装图文教程
2016/05/26 Python
一道python走迷宫算法题
2018/01/22 Python
使用Python实现在Windows下安装Django
2018/10/17 Python
python命令行工具Click快速掌握
2019/07/04 Python
pytorch 模型可视化的例子
2019/08/17 Python
Python3 操作 MySQL 插入一条数据并返回主键 id的实例
2020/03/02 Python
基于HTML5陀螺仪实现ofo首页眼睛移动效果的示例
2017/07/31 HTML / CSS
Bootstrap File Input文件上传组件
2020/12/01 HTML / CSS
时装界的“朋克之母”:Vivienne Westwood
2017/07/06 全球购物
PatPat香港:婴童服饰和亲子全家装在线购物
2020/09/27 全球购物
什么是SQL Server的确定性函数和不确定性函数
2016/08/04 面试题
材料采购员岗位职责
2013/12/17 职场文书
外贸业务员求职信
2014/06/16 职场文书
忠诚与背叛观后感
2015/06/04 职场文书