javascript中的原型链深入理解


Posted in Javascript onFebruary 24, 2014

要弄清楚原型链就要先弄清楚 function 类型,在javascript中没有类的概念,都是函数,所以它是一门函数式的编程语言。类有一个很重要的特性,就是它可以根据它的构造函数来创建以它为模板的对象。在javascript中,函数就有2个功能

第一、 作为一般函数调用
第二、 作为它原型对象的构造函数 也就new()

我们来看一个例子

function a(){ 
this.name = 'a'; 
}

当创建一个函数,它会发生什么呢?

第一、它会创建1个函数对象 也就是a 本身

第二、它会创建1个原型对象@a(用@来表示)

第三、函数对象会有一个prototype指针,它指向了对应的原型对象,这里就指向了@a

第四、@a对象中有一个construtor指针,指向它的构造函数,这里就指向了a

http://img.blog.csdn.net/20140222125611500?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbGpsMTU3MDEx/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast

这个prototype属性究竟有什么用呢?

其实prototype 属性表示当前函数能够控制的范围(或者说它指明了当前函数是谁的构造函数),这里a就是@a原型对象的构造函数,所以我们会看见有这种写法

function a(){ 
this.name = 'a'; 
} var a1 = new a();

这就和其他常见语言相似了,new 就是调用原型对象(通过prototype指针)里面构造函数(constructor)创建一个新的对象实例。

那么修改了prototype指向对象里面的属性,也就影响了所有以它为模板创建的实例,我们可以这样来验证

function a(){ 
this.name = 'a'; 
} var a1 = new a(); 
a.prototype.age = 1; 
alert(a1.age);

结果:1

那为什么a1对象可以直接访问到age属性呢?a1对象里面我并没有定义age属性啊,

那是因为所有实例里面都会有一个引用_proto_(在firfox,chrome下可以直接访问,ie不支持)指向了这个原型,这里就是指向了@a,

function a(){ 
this.name = 'a'; 
} var a1 = new a(); 
alert(a1._proto_ == a.prototype)

结果:true

在访问属性的时候,会先在a1对象内部中寻找,如果没有,就会顺着_proto_指向的对象里面去寻找,这里会到@a中寻找,找到就返回值,没有找到就返回undefined,用个成语来形容,就是顺藤摸瓜嘛!

至此原型链的含义就出来了,由于原型对象也有一个_proto_指针,又指向了另一个原型,一个接一个,就形成了原型链。Object.prototype是最顶层的原型,所以如果修改了Object.prototype的属性,那么就影响了所有的对象。

在来看一段代码

function a(){ 
this.name = 'a'; 
} function b(){ 
this.age = 1; 
} 
b.prototype = new a(); 
alert(new b().name);

我们显示的将b的原型指向了a的一个实例,然后,b的实例也可以访问a的属性了。这就是javascript的继承了,那为什么b.prototype 指向的是a的一个实例,而不是直接指向a.prototype 呢?
b.prototype = new a.prototype;

如果像上面这么写,修改p.prototype中的属性,那么a的原型也会改变了,相当于是子类修改了父类,并且子类和父类的属性糅合在了一起,这显然是不合适的。换句话说,b也成为了@a的构造函数,a,b成了平级的关系。

我们可以下一个定义:

函数a 继承函数b 也就是让函数a成为函数b原型的一个实例的构造函数,构造函数里面声明的属性是函数a自己的,原型实例里面的属性就是继承b的

var $ = jQuery = function(selector,context){ 
//不可能在自己的构造函数中又一次构造自己,所以返回了另外一个构造函数的实例 
return new init(selector,context); 
} 
jQuery.fn = jQuery.prototype = { 
size:function(){ 
return this.length; 
} 
} function init (selector,context){ 
} 
init.prototype = jQuery.fn;; 
}

这是jquery的一段源码,我们在使用jquery的时候,并没有使用new关键字,那它是如何构造对象的呢?

用上面的知识,可以解释,jquery这里只是一个一般函数的调用,它返回了jquery原型的另外一个构造函数创建的对象,也就是new init()

Javascript 相关文章推荐
javascript Array.remove() 数组删除
Aug 06 Javascript
浅说js变量
May 25 Javascript
mailto的使用技巧分享
Dec 21 Javascript
深入理解JavaScript函数参数(推荐)
Jul 26 Javascript
web 前端常用组件之Layer弹出层组件
Sep 22 Javascript
NPM 安装cordova时警告:npm WARN deprecated minimatch@2.0.10: Please update to minimatch 3.0.2 or higher to
Dec 20 Javascript
Javascript中的async awai的用法
May 17 Javascript
原生JS实现图片懒加载(lazyload)实例
Jun 13 Javascript
微信小程序左右滑动的实现代码
Dec 15 Javascript
详解element-ui设置下拉选择切换必填和非必填
Jun 17 Javascript
Layui 动态禁止select下拉的例子
Sep 03 Javascript
如何使用JavaScript检测空闲的浏览器选项卡
May 28 Javascript
JSONP获取Twitter和Facebook文章数的具体步骤
Feb 24 #Javascript
动态加载js、css等文件跨iframe实现
Feb 24 #Javascript
js操纵dom生成下拉列表框的方法
Feb 24 #Javascript
js设置function参数默认值(适合没有传参情况)
Feb 24 #Javascript
JS取request值以及自动执行使用示例
Feb 24 #Javascript
减少访问DOM的次数提升javascript性能
Feb 24 #Javascript
createTextRange()的使用示例含文本框选中部分文字内容
Feb 24 #Javascript
You might like
世界收音机发展史
2021/03/01 无线电
劣质的PHP代码简化
2010/02/08 PHP
php与mysql建立连接并执行SQL语句的代码
2011/07/04 PHP
兼容firefox,chrome的网页灰度效果
2011/08/08 PHP
PHP获取搜索引擎关键字来源的函数(支持百度和谷歌等搜索引擎)
2012/10/03 PHP
php页面函数设置超时限制的方法
2014/12/01 PHP
php通过ksort()函数给关联数组按照键排序的方法
2015/03/18 PHP
PHP链表操作简单示例
2016/10/15 PHP
php实现支持中文的文件下载功能示例
2017/08/30 PHP
phpStudy 2016 使用教程详解(支持PHP7)
2017/10/18 PHP
静态的动态续篇之来点XML
2006/08/15 Javascript
javascript客户端解决方案 缓存提供程序
2010/07/14 Javascript
中文字符串截取的js函数代码
2013/04/17 Javascript
js Map List 遍历使用示例
2013/07/10 Javascript
jquery中对于批量deferred的处理方法
2014/01/22 Javascript
jQuery Validate插件实现表单强大的验证功能
2015/12/18 Javascript
Angularjs 动态添加指令并绑定事件的方法
2017/04/13 Javascript
详解JavaScript调用栈、尾递归和手动优化
2017/06/03 Javascript
JavaScript高级函数应用之分时函数实例分析
2018/08/03 Javascript
Vue.js组件间通信方式总结【推荐】
2018/11/23 Javascript
详解jQuery-each()方法
2019/03/13 jQuery
layui使用及简单的三级联动实现教程
2020/12/01 Javascript
浅谈python之新式类
2018/08/12 Python
Django中使用第三方登录的示例代码
2018/08/20 Python
Python异常处理例题整理
2019/07/07 Python
Django异步任务线程池实现原理
2019/12/17 Python
经济学人订阅:The Economist
2018/07/19 全球购物
英国最好的包装供应商:Priory Direct
2019/12/17 全球购物
Java编程面试题
2016/04/04 面试题
时尚休闲吧创业计划书
2014/01/25 职场文书
房屋公证委托书
2014/04/03 职场文书
网吧消防安全责任书
2014/07/29 职场文书
学习型党组织心得体会
2014/09/12 职场文书
总经理致辞
2015/07/29 职场文书
技术转让协议书
2016/03/19 职场文书
Matlab如何实现矩阵复制扩充
2021/06/02 Python