理解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 相关文章推荐
js 图片随机不定向浮动的实现代码
Jul 02 Javascript
jQuery之字体大小的设置方法
Feb 27 Javascript
HTML页面定时跳转方法解析(2种任选)
Dec 22 Javascript
vue 组件中slot插口的具体用法
Apr 03 Javascript
详解javascript中的babel到底是什么
Jun 21 Javascript
React key值的作用和使用详解
Aug 23 Javascript
js实现文件上传功能 后台使用MultipartFile
Sep 08 Javascript
解决vue axios的封装 请求状态的错误提示问题
Sep 25 Javascript
小程序根据手机机型设置自定义底部导航距离
Jun 04 Javascript
微信小程序-可移动菜单的实现过程详解
Jun 24 Javascript
vue实现一个矩形标记区域(rectangle marker)的方法
Oct 28 Javascript
详解JavaScript中Arguments对象用途
Aug 30 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
php实现复制移动文件的方法
2015/07/29 PHP
浅析php设计模式之数据对象映射模式
2016/03/03 PHP
php封装的smarty类完整实例
2016/10/19 PHP
学习ExtJS Window常用方法
2009/10/07 Javascript
JQuery扩展插件Validate 1 基本使用方法并打包下载
2011/09/05 Javascript
js获取class的所有元素
2013/03/28 Javascript
JavaScript 中的日期和时间及表示标准介绍
2013/08/21 Javascript
jquery 为a标签绑定click事件示例代码
2014/06/23 Javascript
一款由jquery实现的整屏切换特效
2014/09/15 Javascript
jQuery实现点击后高亮背景固定显示的菜单效果【附demo源码下载】
2016/09/21 Javascript
AngularJS表单验证中级篇(3)
2016/09/28 Javascript
javascript 内置对象及常见API详细介绍
2016/11/01 Javascript
谈谈JS中常遇到的浏览器兼容问题和解决方法
2016/12/17 Javascript
bootstrap实现每隔5秒自动轮播效果
2016/12/20 Javascript
angular+ionic 的app上拉加载更新数据实现方法
2017/01/16 Javascript
前端构建工具之gulp的语法教程
2017/06/12 Javascript
利用JS如何计算字符串所占字节数示例代码
2017/09/13 Javascript
js数组去重的N种方法(小结)
2018/06/07 Javascript
vue实现手机端省市区区域选择
2019/09/27 Javascript
vue项目打包之开发环境和部署环境的实现
2020/04/23 Javascript
ant design vue 表格table 默认勾选几项的操作
2020/10/31 Javascript
[02:26]2018DOTA2亚洲邀请赛赛前采访-Newbee篇
2018/04/03 DOTA
[03:11]TI9战队档案 - Alliance
2019/08/20 DOTA
Python实现保证只能运行一个脚本实例
2015/06/24 Python
玩转python爬虫之URLError异常处理
2016/02/17 Python
python xml.etree.ElementTree遍历xml所有节点实例详解
2016/12/04 Python
目前最全的python的就业方向
2018/06/05 Python
django序列化serializers过程解析
2019/12/14 Python
Python3.9.1中使用match方法详解
2021/02/08 Python
HTML5 Web 存储详解
2016/09/16 HTML / CSS
canvas实现有递增动画的环形进度条的实现方法
2019/07/10 HTML / CSS
公司廉洁自律承诺书
2014/03/27 职场文书
挂靠协议书范本
2014/04/22 职场文书
供应商食品安全承诺书
2015/04/29 职场文书
2015年小学重阳节活动总结
2015/07/29 职场文书
一文搞懂Golang 时间和日期相关函数
2021/12/06 Golang