理解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在IE或Firefox下获取鼠标位置的代码
Dec 18 Javascript
jquery isType() 类型判断代码
Feb 14 Javascript
js获取控件位置以及不同浏览器中的差别介绍
Aug 08 Javascript
JavaScript利用append添加元素报错的解决方法
Jul 01 Javascript
JS实现判断碰撞的方法
Feb 11 Javascript
jquery实现页面百叶窗走马灯式翻滚显示效果的方法
Mar 12 Javascript
jQuery层动画定位滑动效果的方法
Apr 30 Javascript
AngularJS 面试题集锦
Sep 06 Javascript
Bootstrap禁用响应式布局的实现方法
Mar 09 Javascript
JS实现的对象去重功能示例
Jun 04 Javascript
vue 限制input只能输入正数的操作
Aug 05 Javascript
ES11新增的这9个新特性,你都掌握了吗
Oct 15 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基本语法总结
2014/09/06 PHP
php连接odbc数据源并保存与查询数据的方法
2014/12/24 PHP
php绘图之生成饼状图的方法
2015/01/24 PHP
php反射学习之不用new方法实例化类操作示例
2019/06/14 PHP
关于Anemometer图形化显示MySQL慢日志的工具搭建及使用的详细介绍
2020/07/13 PHP
仅用[]()+!等符号就足以实现几乎任意Javascript代码
2010/03/01 Javascript
JavaScript 开发规范要求(图文并茂)
2010/06/11 Javascript
jQuery使用数组编写图片无缝向左滚动
2012/12/11 Javascript
JavaScript获取/更改文本框的值的实例代码
2013/08/02 Javascript
JS实现清除指定cookies的方法
2014/09/20 Javascript
JavaScript动态插入CSS的方法
2015/12/10 Javascript
浅谈addEventListener和attachEvent的区别
2016/07/14 Javascript
浅谈js常用内置方法和对象
2016/09/24 Javascript
原生JS版和jquery版实现checkbox的全选/全不选/点选/行内点选(Mr.Think)
2016/10/29 Javascript
angularjs 实现带查找筛选功能的select下拉框实例
2017/01/11 Javascript
JavaScript闭包的简单应用
2017/09/01 Javascript
react 实现页面代码分割、按需加载的方法
2018/04/03 Javascript
深入理解JavaScript和TypeScript中的class
2018/04/22 Javascript
Javascript获取某个月的天数
2018/05/30 Javascript
如何将HTML字符转换为DOM节点并动态添加到文档中详解
2018/08/19 Javascript
Django模板继承 extend标签实例代码详解
2019/05/16 Javascript
使用vue自定义指令开发表单验证插件validate.js
2019/05/23 Javascript
[59:00]DOTA2-DPC中国联赛 正赛 Ehome vs PSG.LGD BO3 第一场 3月7日
2021/03/11 DOTA
python 创建弹出式菜单的实现代码
2017/07/11 Python
python实现输入的数据在地图上生成热力图效果
2019/12/06 Python
查看端口并杀进程python脚本代码
2019/12/17 Python
Tensorflow训练MNIST手写数字识别模型
2020/02/13 Python
解决Jupyter notebook更换主题工具栏被隐藏及添加目录生成插件问题
2020/04/20 Python
CSS3 display知识详解
2015/11/25 HTML / CSS
亚马逊西班牙购物网站:amazon西班牙
2017/03/06 全球购物
联想马亚西亚官方网站:Lenovo Malaysia
2018/09/19 全球购物
实习协议书范本
2014/04/22 职场文书
蛋糕店创业计划书
2014/05/06 职场文书
国际贸易系求职信
2014/08/09 职场文书
关于运动会的广播稿(10篇)
2014/09/12 职场文书
用Python编写简单的gRPC服务的详细过程
2021/07/04 Python