理解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实现预览待上传的本地图片
Mar 15 Javascript
jQuery筛选器children()案例详解(图文)
Feb 17 Javascript
使用GruntJS构建Web程序之合并压缩篇
Jun 06 Javascript
jQuery实现鼠标滑过点击事件音效试听
Aug 31 Javascript
JS+CSS实现经典的左侧竖向滑动菜单效果
Sep 23 Javascript
Ionic实现页面下拉刷新(ion-refresher)功能代码
Jun 03 Javascript
Node.js dgram模块实现UDP通信示例代码
Sep 26 Javascript
jQuery图片加载失败替换默认图片方法汇总
Nov 29 jQuery
解决vue中监听input只能输入数字及英文或者其他情况的问题
Aug 30 Javascript
微信小程序CSS3动画下拉菜单效果
Nov 04 Javascript
浅谈vue中document.getElementById()拿到的是原值的问题
Jul 26 Javascript
openlayers4.6.5实现距离量测和面积量测
Sep 25 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 file_exists无效的解决办法
2013/06/26 PHP
destoon整合UCenter图文教程
2014/06/21 PHP
PHP+jQuery 注册模块的改进(三):更新到Smarty3.1
2014/10/14 PHP
Yii2 ActiveRecord多表关联及多表关联搜索的实现
2016/06/30 PHP
基于ThinkPHP5.0实现图片上传插件
2017/09/25 PHP
PHP+fiddler抓包采集微信文章阅读数点赞数的思路详解
2019/12/20 PHP
Yii框架组件的事件机制原理与用法分析
2020/04/07 PHP
分享20款好玩的jQuery游戏
2011/04/17 Javascript
js静态方法与实例方法分析
2011/07/04 Javascript
调试Javascript代码(浏览器F12及VS中debugger关键字)
2013/01/25 Javascript
百度移动版的url编码解码示例
2014/04/29 Javascript
让浏览器DOM元素最后加载的js方法
2014/07/29 Javascript
jQuery中html()方法用法实例
2014/12/25 Javascript
以JavaScript来实现WordPress中的二级导航菜单的方法
2015/12/14 Javascript
解决拦截器对ajax请求的拦截实例详解
2016/12/21 Javascript
微信小程序 支付功能(前端)的实现
2017/05/24 Javascript
JS实现多张图片预览同步上传功能
2017/06/23 Javascript
微信小程序 密码输入(源码下载)
2017/06/27 Javascript
Vue父子组件双向绑定传值的实现方法
2018/07/31 Javascript
js动态设置select下拉菜单的默认选中项实例
2018/08/21 Javascript
vue.js 实现点击按钮动态添加li的方法
2018/09/07 Javascript
使用 Python 实现微信群友统计器的思路详解
2018/09/26 Python
python实现根据指定字符截取对应的行的内容方法
2018/10/23 Python
漂亮的Django Markdown富文本app插件的实现
2019/01/02 Python
Tkinter中复选菜单是否被选中的判断与设置方式
2020/03/04 Python
python调用私有属性的方法总结
2020/07/24 Python
python切割图片的示例
2020/11/12 Python
CSS3教程(10):CSS3 HSL声明设置颜色
2009/04/02 HTML / CSS
css3和jquery实现的可折叠导航菜单适合放在手机网页的导航菜单
2014/09/02 HTML / CSS
澳大利亚领先的在线机械五金、园艺和存储专家:Edisons
2018/03/24 全球购物
中学生运动会入场词
2014/02/12 职场文书
《鞋匠的儿子》教学反思
2014/03/02 职场文书
国庆节主题班会
2015/08/15 职场文书
教你快速开启Apache SkyWalking的自监控
2021/04/25 Servers
利用python进行数据加载
2021/06/20 Python
SpringBoot2零基础到精通之数据库专项精讲
2022/03/22 Java/Android