理解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 相关文章推荐
javascript web对话框与弹出窗口
Feb 22 Javascript
javascript event 事件解析
Jan 31 Javascript
javascript的数据类型、字面量、变量介绍
May 23 Javascript
利用Keydown事件阻止用户输入实现代码
Mar 11 Javascript
jQuery满屏焦点图左右滚动特效代码分享
Sep 07 Javascript
一个仿微博登陆邮箱提示框js开发案例
Jul 28 Javascript
Vue.js中数组变动的检测详解
Oct 12 Javascript
浅析js中mvvm模式实现的原理
Oct 06 Javascript
Vue 理解之白话 getter/setter详解
Apr 16 Javascript
VUE前后端学习tab写法实例
Aug 06 Javascript
解决Vue的项目使用Element ui 走马灯无法实现的问题
Aug 03 Javascript
Echarts如何重新渲染实例详解
May 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+MySQL5.0中文乱码解决方法
2006/11/20 PHP
如何使用PHP计算上一个月的今天
2013/05/23 PHP
详解PHP防止盗链防止迅雷下载的方法
2017/04/26 PHP
jQuery实现 注册时选择阅读条款 左右移动
2013/04/11 Javascript
JavaScript 函数惰性载入的实现及其优点介绍
2013/08/12 Javascript
JavaScript实现穷举排列(permutation)算法谜题解答
2014/12/29 Javascript
JavaScript插件化开发教程 (二)
2015/01/27 Javascript
学习使用grunt来打包JavaScript和CSS程序的教程
2016/01/04 Javascript
JS控制静态页面传递参数并获取参数应用
2016/08/10 Javascript
js分页之前端代码实现和请求处理
2017/08/04 Javascript
Node.js pipe实现源码解析
2017/08/12 Javascript
微信小程序实现动态设置页面标题的方法【附源码下载】
2017/11/29 Javascript
微信小程序网络请求封装示例
2018/07/24 Javascript
angular将html代码输出为内容的实例
2018/09/30 Javascript
微信小程序时间控件picker view使用详解
2018/12/28 Javascript
Javascript异步执行不按顺序解决方案
2020/04/30 Javascript
完美解决通过IP地址访问VUE项目的问题
2020/07/18 Javascript
[01:48]帕吉至宝加入游戏,遗迹战场现“千劫神屠”
2018/04/07 DOTA
对Python新手编程过程中如何规避一些常见问题的建议
2015/04/01 Python
python对url格式解析的方法
2015/05/13 Python
python实现烟花小程序
2019/01/30 Python
Python学习笔记之文件的读写操作实例分析
2019/08/07 Python
Nginx+Uwsgi+Django 项目部署到服务器的思路详解
2020/05/08 Python
python 实现Harris角点检测算法
2020/12/11 Python
css3 伪元素和伪类选择器详解
2014/09/04 HTML / CSS
美国指甲油品牌:Deco Miami
2017/01/30 全球购物
摩飞电器俄罗斯官方网站:Morphy Richards俄罗斯
2020/07/30 全球购物
房地产端午节活动方案
2014/08/24 职场文书
门卫岗位职责
2015/02/09 职场文书
2015年档案室工作总结
2015/05/23 职场文书
2016年记者节感言
2015/12/08 职场文书
高三英语教学反思
2016/03/03 职场文书
Python词云的正确实现方法实例
2021/05/08 Python
浅谈CSS不规则边框的生成方案
2021/05/25 HTML / CSS
python数字类型和占位符详情
2022/03/13 Python
React如何使用axios请求数据并把数据渲染到组件
2022/08/05 Javascript