理解Javascript_05_原型继承原理


Posted in Javascript onOctober 13, 2010

prototype与[[prototype]]

在有面象对象基础的前提下,来看一段代码:

//Animal构造函数 
function Animal(name){ 
this.name = name; 
} 
//Animal原型对象 
Animal.prototype = { 
id:"Animal", 
sleep:function(){ 
alert("sleep"); 
} 
} var dog = new Animal("旺才"); 
alert(dog.name);//旺才 
alert(dog.id);//Animal 
dog.sleep()//sleep

其对应的简易内存分配结构图:
理解Javascript_05_原型继承原理
现在让我们来解释一下这张内存图的来龙去脉:

首先明确一点[[prototype]]与prototype并不是同一个东西。

那先来看prototype,每一个函数对象都有一个显示的prototype属性,它代表了对象的原型,更明确的说是代表了由函数对象(构造函数)所创建出来的对象的原型。结合本例,Animal.prototype就是dog的原型,dog所引用的那个对象将从Animal.prototype所引用的对象那继承属性与方法。

每个对象都有一个名为[[Prototype]]的内部属性,指向于它所对应的原型对象。在本例中dog的[[prototype]]指向Animal.prototype,大家都知道,Animal.prototype也是一个对象,即然是一个对象,那它必然也有[[prototype]]属性指向于它所对应的原型对象,由此便构成了一种链表的结构,这就是原型链的概念。额外要说的是:不同的JS引擎实现者可以将内部[[Prototype]]属性命名为任何名字,并且设置它的可见性,前且只在JS引擎内部使用。虽然无法在JS代码中访问到内部[[Prototype]](FireFox中可以,名字为__proto__因为Mozilla将它公开了),但可以使用对象的 isPrototypeOf()方法进行测试,注意这个方法会在整个Prototype链上进行判断。

注:关于函数对象的具体内容,将在后继的博文中讲解。

属性访问原则

使用obj.propName访问一个对象的属性时,按照下面的步骤进行处理(假设obj的内部[[Prototype]]属性名为__proto__):
1. 如果obj存在propName属性,返回属性的值,否则
2. 如果obj.__proto__为null,返回undefined,否则
3. 返回obj.__proto__.propName
调用对象的方法跟访问属性搜索过程一样,因为方法的函数对象就是对象的一个属性值。
提示: 上面步骤中隐含了一个递归过程,步骤3中obj.__proto__是另外一个对象,同样将采用1, 2, 3这样的步骤来搜索propName属性。
理解Javascript_05_原型继承原理
这就是基于Prototype的继承和共享。其中object1的方法fn2来自object2,概念上即object2重写了object3的方法fn2。
JavaScript对象应当都通过prototype链关联起来,最顶层是Object,即对象都派生自Object类型。

结合是上面的理论,让我们再来看一个更加复杂的示例,他明确的解释了prototype、[[prototype]]、原型链以及属性访问的相关要点:

//Animal构造函数 
function Animal(name){ 
this.name = name; 
} 
//Animal原型对象 
Animal.prototype = { 
id:"Animal", 
sleep:function(){ 
alert("sleep"); 
} 
} function Human(name,age){ 
Animal.call(this,name); 
this.age = age; 
} 
Human.prototype = new Animal(); 
Human.prototype.id = "Human"; 
Human.prototype.say = function(){ 
alert("hello everyone,My name is "+this.name +",I'm "+this.age+" and I'm a "+this.id); 
} 
//Human相关调用 
var jxl = new Human('笨蛋',25); 
alert(jxl.name);//笨蛋 
alert(jxl.id);//Human 
jxl.say();//hello everyone,My name is 笨蛋,I'm 25 and I'm a Human 
alert(Animal.prototype.isPrototypeOf(jxl));//true 
alert(Object.prototype.isPrototypeOf(jxl));//true

根据上面的代码,你能画出相应的内存图吗?好,让我们来看一下:
理解Javascript_05_原型继承原理
注:prototype的根为Object.prototype,对象Object.prototype的内部[[prototype]]属性为null.
其实,这里还有很多东西可以讲,但在其原理都在这张图上了,可试着调整一下代码的次序,如将Human.prototype.id = "Human";放在Human.prototype = new Animal();的前面,看一下运行结果,解释一下为什么之类的,你可以学到很多。

我发现,通过内存来展现程序内部运行细节真的是太完美了!

Javascript 相关文章推荐
对xmlHttp对象方法和属性的理解
Jan 17 Javascript
jquery显示和隐藏div特效实例
Feb 27 Javascript
将HTML的左右尖括号等转义成实体形式的两种实现方式
May 04 Javascript
使用jQuery监听DOM元素大小变化
Feb 24 Javascript
JavaScript操作选择对象的简单实例
May 16 Javascript
webpack入门+react环境配置
Feb 08 Javascript
微信小程序之数据双向绑定与数据操作
May 12 Javascript
解决IE7中使用jQuery动态操作name问题
Aug 28 jQuery
vue 挂载路由到头部导航的方法
Nov 13 Javascript
vue项目设置scrollTop不起作用(总结)
Dec 21 Javascript
详解实现一个通用的“划词高亮”在线笔记功能
Apr 23 Javascript
JavaScript使用localStorage存储数据
Sep 25 Javascript
JavaScript 打地鼠游戏代码说明
Oct 12 #Javascript
理解Javascript_03_javascript全局观
Oct 11 #Javascript
理解Javascript_02_理解undefined和null
Oct 11 #Javascript
理解Javascript_01_理解内存分配原理分析
Oct 11 #Javascript
javascript getElementsByClassName实现代码
Oct 11 #Javascript
javascript Array.prototype.slice使用说明
Oct 11 #Javascript
javascript 伪数组实现方法
Oct 11 #Javascript
You might like
CentOS 安装 PHP5.5+Redis+XDebug+Nginx+MySQL全纪录
2015/03/25 PHP
PHP使用内置函数生成图片的方法详解
2016/05/09 PHP
网页图片延时加载的js代码
2010/04/22 Javascript
javascript学习笔记(九) js对象 设计模式
2012/06/19 Javascript
jQuery源码中的chunker 正则过滤符分析
2012/07/31 Javascript
Jquery显示、隐藏元素以及添加删除样式
2013/08/09 Javascript
Js base64 加密解密介绍
2013/10/11 Javascript
JS获取URL中的参数数据
2013/12/05 Javascript
jQuery中nextAll()方法用法实例
2015/01/07 Javascript
Bootstrap表单布局样式代码
2016/05/31 Javascript
Bootstrap源码解读媒体对象、列表组和面板(10)
2016/12/26 Javascript
javascript实现秒表计时器的制作方法
2017/02/16 Javascript
webpack配置的最佳实践分享
2017/04/21 Javascript
神级程序员JavaScript300行代码搞定汉字转拼音
2017/05/20 Javascript
详解vue中computed 和 watch的异同
2017/06/30 Javascript
node+koa实现数据mock接口的方法
2017/09/20 Javascript
微信小程序使用audio组件播放音乐功能示例【附源码下载】
2017/12/08 Javascript
探索Vue高阶组件的使用
2018/01/08 Javascript
jQuery实现为table表格动态添加或删除tr功能示例
2019/02/19 jQuery
Js生成随机数/随机字符串的方法小结【5种方法】
2020/05/27 Javascript
跟老齐学Python之不要红头文件(2)
2014/09/28 Python
基于python编写的微博应用
2014/10/17 Python
Python实现基于TCP UDP协议的IPv4 IPv6模式客户端和服务端功能示例
2018/03/22 Python
Flask框架实现给视图函数增加装饰器操作示例
2018/07/16 Python
python实现在图片上画特定大小角度矩形框
2018/10/24 Python
对Python 获取类的成员变量及临时变量的方法详解
2019/01/22 Python
python读取hdfs上的parquet文件方式
2020/06/06 Python
Python私有属性私有方法应用实例解析
2020/09/15 Python
Python提取视频中图片的示例(按帧、按秒)
2020/10/22 Python
玩具反斗城天猫官方旗舰店:享誉全球的玩具店
2017/10/10 全球购物
美国最大最全的亚洲购物网站:美国亚米网(Yamibuy)
2020/05/05 全球购物
中专毕业生自我鉴定
2014/02/02 职场文书
服装店营销方案
2014/03/10 职场文书
科研课题实施方案
2014/03/18 职场文书
三八妇女节致辞
2015/07/31 职场文书
学校就业保障协议书
2019/06/24 职场文书