理解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 相关文章推荐
jQuery 获取对象 基本选择与层级
May 31 Javascript
js工具方法弹出蒙版
May 08 Javascript
深入理解javascript原型链和继承
Sep 23 Javascript
JavaScript控制网页平滑滚动到指定元素位置的方法
Apr 17 Javascript
js实现选中页面文字将其分享到新浪微博
Nov 05 Javascript
Treegrid的动态加载实例代码
Apr 29 Javascript
webpack配置打包后图片路径出错的解决
Apr 26 Javascript
20个最常见的jQuery面试问题及答案
May 23 jQuery
JavaScript对象拷贝与Object.assign用法实例分析
Jun 20 Javascript
node.js学习笔记之koa框架和简单爬虫练习
Dec 13 Javascript
微信小程序常见页面跳转操作简单示例
May 01 Javascript
Vue项目页面跳转时浏览器窗口上方显示进度条功能
Mar 26 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
Zend Framework教程之Zend_Form组件实现表单提交并显示错误提示的方法
2016/03/21 PHP
PHP错误处理函数register_shutdown_function使用示例
2017/07/03 PHP
php7 新增功能实例总结
2020/05/25 PHP
javaScript 数值型和字符串型之间的转换
2009/07/25 Javascript
javascript json2 使用方法
2010/03/16 Javascript
JS trim去空格的最佳实践
2011/10/30 Javascript
js 窗口抖动示例
2013/09/04 Javascript
JS实现匀速运动的代码实例
2013/11/29 Javascript
JS给超链接加确认对话框的方法
2015/02/24 Javascript
js实现仿百度汽车频道选择汽车图片展示实例
2015/05/06 Javascript
Angularjs实现多个页面共享数据的方式
2016/03/29 Javascript
JavaScript和JQuery获取DIV值的方法示例
2017/03/07 Javascript
Node.js和Express简单入门介绍
2017/03/24 Javascript
Vue-resource拦截器判断token失效跳转的实例
2017/10/27 Javascript
json字符串传到前台input的方法
2018/08/06 Javascript
vue 使用html2canvas将DOM转化为图片的方法
2018/09/11 Javascript
微信小程序用户拒绝授权的处理方法详解
2019/09/20 Javascript
在vue项目中引用Antv G2,以饼图为例讲解
2020/10/28 Javascript
[02:28]DOTA2英雄基础教程 灰烬之灵
2013/12/19 DOTA
rabbitmq(中间消息代理)在python中的使用详解
2017/12/14 Python
Python爬虫爬取一个网页上的图片地址实例代码
2018/01/16 Python
Python测试网络连通性示例【基于ping】
2018/08/03 Python
使用Python脚本zabbix自定义key监控oracle连接状态
2019/08/28 Python
DjangoWeb使用Datatable进行后端分页的实现
2020/05/18 Python
在Keras中利用np.random.shuffle()打乱数据集实例
2020/06/15 Python
Python安装并操作redis实现流程详解
2020/10/13 Python
使用Canvas操作像素的方法
2018/06/14 HTML / CSS
VIVOBAREFOOT赤脚鞋:让您的脚做自然的事情
2017/06/01 全球购物
俄罗斯最大的消费电子连锁零售商:Mvideo
2017/06/25 全球购物
链表面试题-一个链表的结点结构
2015/05/04 面试题
党员2014两会学习心得体会
2014/03/17 职场文书
525心理活动总结
2014/07/04 职场文书
人事代理委托书
2014/09/27 职场文书
一个都不能少观后感
2015/06/04 职场文书
vue+spring boot实现校验码功能
2021/05/27 Vue.js
Html5大屏数据可视化开发的实现
2021/06/11 HTML / CSS