理解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 相关文章推荐
IE iframe的onload方法分析小结
Jan 07 Javascript
不用锚点也可以平滑滚动到页面的指定位置实现代码
May 08 Javascript
jquery validate表单验证的基本用法入门
Jan 18 Javascript
解析JavaScript实现DDoS攻击原理与保护措施
Dec 26 Javascript
完美解决node.js中使用https请求报CERT_UNTRUSTED的问题
Jan 08 Javascript
js实现年月日表单三级联动
Apr 17 Javascript
Vue.js 中的 $watch使用方法
May 25 Javascript
详细分析JS函数去抖和节流
Dec 05 Javascript
Angular5.0 子组件通过service传递值给父组件的方法
Jul 13 Javascript
vue canvas绘制矩形并解决由clearRec带来的闪屏问题
Sep 02 Javascript
微信小程序自定义顶部组件customHeader的示例代码
Jun 03 Javascript
Element Steps步骤条的使用方法
Jul 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
ADODB类使用
2006/11/25 PHP
Yii框架在页面输出执行sql语句以方便调试的实现方法
2014/12/24 PHP
php随机生成数字字母组合的方法
2015/03/18 PHP
php实现算术验证码功能
2018/12/05 PHP
php的无刷新操作实现方法分析
2020/02/28 PHP
php的lavarel框架中join和orWhere的用法
2020/12/28 PHP
调试Node.JS的辅助工具(NodeWatcher)
2012/01/04 Javascript
$.get获取一个文件的内容示例代码
2013/09/11 Javascript
js onclick事件传参讲解
2013/11/06 Javascript
如何获取select下拉框的值(option没有及有value属性)
2013/11/08 Javascript
JavaScript charCodeAt方法入门实例(用于取得指定位置字符的Unicode编码)
2014/10/17 Javascript
JS动态改变浏览器标题的方法
2016/04/06 Javascript
AngularJS学习笔记之依赖注入详解
2016/05/16 Javascript
AngularJS 单选框及多选框的双向动态绑定
2017/04/20 Javascript
ztree实现权限横向显示功能
2017/05/20 Javascript
利用canvas实现的加载动画效果实例代码
2017/07/05 Javascript
基于Vue生产环境部署详解
2017/09/15 Javascript
浅析Vue项目中使用keep-Alive步骤
2018/07/27 Javascript
vue点击按钮动态创建与删除组件功能
2019/12/29 Javascript
谈谈我在vue-cli3中用预渲染遇到的坑
2020/04/22 Javascript
基于JQuery实现页面定时弹出广告
2020/05/08 jQuery
vue中template的三种写法示例
2020/10/21 Javascript
[01:04:30]Fnatic vs Mineski 2018国际邀请赛小组赛BO2 第二场 8.17
2018/08/18 DOTA
Python中Iterator迭代器的使用杂谈
2016/06/20 Python
Python创建xml文件示例
2017/03/22 Python
python实现微信防撤回神器
2019/04/29 Python
python之yield和Generator深入解析
2019/09/18 Python
解决pycharm启动后总是不停的updating indices...indexing的问题
2019/11/27 Python
Python操作MongoDb数据库流程详解
2020/03/05 Python
Pycharm编辑器功能之代码折叠效果的实现代码
2020/10/15 Python
python 实现音频叠加的示例
2020/10/29 Python
中国专业的综合网上购物商城:京东
2016/08/02 全球购物
英国最大的电子产品和家电零售企业:Currys PC World
2016/09/24 全球购物
I.T中国官网:精选时尚设计师单品网购平台
2018/03/26 全球购物
什么是就业协议书
2014/04/17 职场文书
Golang ort 中的sortInts 方法
2022/04/24 Golang