JavaScript中原型链的那些事
引言
在面向对象的语言中继承是非常重要的概念,许多面向对象语言都支持两种继承方式:接口继承和实现继承。接口继承制只继承方法签名,而实现继承继承实际的方法。在ECMAScript中函数没有签名,所以ECMAScript无法实现接口继承,只能实现实现继承。那么是怎么实现实现继承的呢??这就要说一说JS中的原型链了。
原型链的定义
什么是原型链?这个问题很简单,其基本思想就是利用原型让一个引用类型继承另一个引用类型的属性和方法。
我们先来回顾一下构造函数,原型,实例之间的关系。每一个构造函数都有一个原型对象,原型对象中包含一个指向构造函数的指针,而实例都包含一个指向原型对象的内部指针。在原型对象中通过prototype指向构造函数,而在实例中通过__proto__指向原型对象,但是该属性是区分浏览器的,这是部分浏览器为实例对象添加的属性,在ECMAScript中表现为[[prototype]]。
现在我们已经知道了原型对象中存在一个指针指向构造函数,现在我们让原型对象等于另一个类型的实例,此时的原型对象将包含一个指向另一个原型的指针,那么另一个原型中也包含一个指向另一个构造函数的指针。加入另一个原型有事另一个类型的实例,那么如此层层递进,就构成了实例与原型的链条。这就是原型链的基本概念。
我的理解:在我看来,我们可以将原型链理解为一个单链表,每一个原型对象都包含一个指向另一个原型对象的指针,如此递进的链接起来,形式单链表(但并不是说原型链的结构就是单链表,这样只是便于理解)。
function SuperType() {
this.property = true;
} Super.prototype.getSuperValue = function() {
return this.property;
} function SubType() {
this.subproperty = false;
} SubType.prototype = new SuperType(); SubType.prototype.getSubValue = function() {
return this.subproperty;
} var instance = new SubType();
console.log(instance.getSuperValue()); //输出:true
可以看出来,在上述代码中,原型链的继承是通过创建SuperType的实例并将实例赋给SubType的原型实现的。本质就是重写原型对象,换成一个新类型的实例。原来存在于SuperType的实例中的所有属性和方法都会存在于SubType.prototype中。最终结果是这样的:instance指向SubType的原型,SubType的原型又指向SuperType的原型。
有一点需要注意的是,instance.constructor现在指向的不是SubType,而是SuperType,原因是SubType的原型指向了另外一个对象SuperType的原型,而这个原型对象的constructor属性指向的是SuperType。
原型搜索机制
通过原型链的实现扩展了原型的原型搜索机制。原型搜索机制就是以读写模式访问一个实例属性时,首先会在实例中搜索该属性,如果没有就会继续搜索实例的原型。通过原型链继承的情况下,搜索过程就会沿着原型链继续向上。首先会搜索实例,在实例中查找是否有需要访问的属性,如果没有,将会搜索实例的原型,看原型中是否有定义的原型属性,如果没有将会通过原型链向上找指向的原型,在找不到属性或方法的情况下回一环一环的到原型链的末端才会停止。
prototype和__proto__的区别
在学习原型链的时候经常搞不懂prototype和__proto__的区别,所以把这两个东西的比较摘出来写成一块。
__proto__属性的来历:创建了自定义的构造函数后其原型对象只会取得constructor属性,其他的方法都是从Object继承的,当使用构造函数的创建一个新的实例的时候该实例内部包含一个指针指向构造函数的原型对象。在ECMAScript中管这个指针叫做[[Prototype]]。在脚本中没有标准的方式访问这个指针。但Firefox、Safari、Chrome在每个对象上都支持一个属性__proto__;但是在其他的实现中,这个属性对脚本是完全不可见的。
默认的原型
所有引用类型默认继承Object,而这个继承也是通过原型链实现的。所有函数的默认原型都是Object的实例,所以在默认原型都会包含一个内部指针,指向Object.prototype。这也是自定义类型都会竭诚toString()、valueOf()等默认方法的原因
谨慎定义方法
1. 给原型添加方法的代码一定要放在替换原型的语句之后。
如下例:
function SuperType() {
this.property = true;
} SuperType.prototype.getSuperValue = function() {
return this.property;
} function SubType() {
this.subpeoperty = false;
} SubType.prototype = new SuperType(); //添加新方法
SubType.prototype.getSubValue = function() {
return this.subpeoperty;
} //重写超类型中的方法
SubType.prototype.getSuperValue = function() {
return false;
} var instance = new SubType();
console.log(instance.getSuperValue()); //输出:false
在上面代码中,重写的方法会屏蔽原来的方法。当通过SubType的实例调用getSuperValue()时,调用的就是重新定义的方法,但通过SuperType的实例调用getSuperValue()时,还会调用原来的方法。
2. 在通过原型链实现继承的时候,不能使用对象字面量创建原型方法。
这样会重写原型链。如下例所示:
function SuperType() {
this.property = true;
} SuperType.prototype.getSuperValue = function() {
return this.property;
} function SubType() {
this.subproperty = false;
} SubType.prototype = new SuperType(); SubType.prototype = {
getSubValue:function() {
return this.subproperty;
}, someOtherMethod: function() {
return false;
}
} var instace = new SubType();
console.log(instace.getSuperValue());
输出:
在上面的例子中,我们把SuperType的实例赋值给原型,紧接着有奖原型替换成一个对象字面量,由于现在的原型包含的是一个Object实例,而非SuperType的实例,一次原型链已经被切断,SuperType和SubType已经没有关系了。
原型链的问题
1. 我们都只知道引用类型的对象中存储的是指向堆内存的指针,所以包含引用类型值的原型属性会被所有实例共享。因为在原型对象中的引用类型只是一个指针,在实例化对象的时候,指针复制,但是指针指向没有发生变化。这也是为什么要在构造函数中,而不是在原型对象中定义属性的原因了。看下面的代码:
function SuperType() {
this.colors = ['red', 'blue', 'green'];
} function SubType() { } SubType.prototype = new SuperType(); var instace1 = new SubType();
instace1.colors.push('black');
console.log(instace1.colors); var instace2 = new SubType();
console.log(instace2.colors); //输出:
// ["red", "blue", "green", "black"]
// ["red", "blue", "green", "black"]
需要注意的是,在JS中基本类型值的原型属性并不是这样的:
function SuperType() {
this.property = true;
} function SubType() { } SubType.prototype = new SuperType(); var instace1 = new SubType();
instace1.property = false;
console.log(instace1.property); var instace2 = new SubType();
console.log(instace2.property); //输出:
// false
// true
原因相比通过上面的实例大家都知道了,在JS中基本类型值的存储并不是通过指针。
2.在创建子类型的实例时,不能向超类型的构造函数中传递参数。
基于这些问题,在实践中我们会很少单独使用原型链,至于怎么在实践中更好地使用原型链,下一篇博客我会详细讲解。
以上~~
JavaScript中原型链的那些事的更多相关文章
- javascript中原型链与instanceof 原理
instanceof:用来判断实例是否是属于某个对象,这个判断依据是什么呢? 首先,了解一下javascript中的原型继承的基础知识: javascript中的对象都有一个__proto__属性,这 ...
- javascript中原型链存在的问题
我们知道使用原型链实现继承是一个goodway:)看个原型链继承的例子. function A () { this.abc = 44; } A.prototype.getAbc = function ...
- JavaScript中原型链存在的问题解析
我们知道使用原型链实现继承是一个goodway:)看个原型链继承的例子. function A () { this.abc = 44; } A.prototype.getAbc = function ...
- 从问题入手,深入了解JavaScript中原型与原型链
从问题入手,深入了解JavaScript中原型与原型链 前言 开篇之前,我想提出3个问题: 新建一个不添加任何属性的对象为何能调用toString方法? 如何让拥有相同构造函数的不同对象都具备相同的行 ...
- javascript的原型链那些事
如果你对javascript的原型链还有任何疑问,请看这篇文章 进入主题 前言 原型链的规则不百分百适用于所有情况 显式原型:prototype,是一个对象{} 隐式原型:__proto__,是一个对 ...
- JavaScript的原型链继承__propt__、prototype、constructor的理解、以及他们之间相互的关系。
回想自己已经工作了有一段时间了,但是自己对JavaScript的原型链.和继承的理解能力没有到位,最近他们彻底的整理并且复习了一遍. 本案例中部分文案来自网络和书籍,如有侵权请联系我,我只是把我的理解 ...
- Javascript的原型链图
90%的前端或者js程序员或者老师们对Javascript懂得不比这个多 给手机看的 但是这个图里的所有褐色单向箭头链就是Javascript的原型链(颜色标注对理解js原型链很关键) 这图中的各个_ ...
- javaScript系列 [04]-javaScript的原型链
[04]-javaScript的原型链 本文旨在花很少的篇幅讲清楚JavaScript语言中的原型链结构,很多朋友认为JavaScript中的原型链复杂难懂,其实不然,它们就像树上的一串猴子. 1.1 ...
- javascript prototype原型链的原理
javascript prototype原型链的原理 说到prototype,就不得不先说下new的过程. 我们先看看这样一段代码: <script type="text/javasc ...
随机推荐
- db2 SQL6036N解决办法
问题背景: 数据库在进行大量的运算和数据处理的过程中,IO.CPU等资源消耗非常高的时候,强制停止数据库.db2stop force 结果数据库命令迟迟没有响应.这个时候对数据库进行其他操作均无响应, ...
- list 删除元素
### List 删除元素 我们以一个字符串为元素类型的 list 为例,进行列表元素的删除: >>> l = ['a', 'b'] 法一:remove(val) 元素值 > ...
- Android 二维码扫描/生成
先看看实现效果 1.在module的build.gradle中执行compile操作 compile 'cn.yipianfengye.android:zxing-library:2.2' 2.在Ap ...
- Python开发【第八篇】:socket网络编程
服务端: import socket server = socket.socket() #绑定要监听的端口 server.bind(('localhost',6969)) #监听 server.lis ...
- js限制图片大小、点击放大图片、点击在新开页面显示
缩放图片到合适大小 function ResizeImages() { var myimg, oldwidth, oldheight; var ...
- vue项目如何通过前端实现自动识别并配置服务器环境地址
前言: 一般来说,一个web项目的生产环境和测试环境的服务器地址一旦确定下来,很少会频繁变动的.那么就可以单独写一个脚本文件,通过当前访问的域名来判断当前的访问环境,然后再通过一定的规则获取对应的服务 ...
- 【网络编程】服务端产生大量的close_wait状态的进程分析
首先要明白close_wait状态是在tcp通信四次握手时的一个中间状态: 即当被动关闭方发送完ACK后进入的状态.这个状态的结束,即要达到下一个状态LASK_ACK需要在发无端发送完剩余的数据后(s ...
- [原]CentOS 7.2 1511部署L2TP/IPsec服务器及客户端
快过年了,感觉从去年开始,我们公司就变成了“别人的公司”,基本上提前一星期就放假了,好开心.正好可以利用这一段时间,把前段时间一些疑惑的问题解决下:) 然而挡在面前的一个拦路虎是:很多时候不能愉快的G ...
- Linux驱动之按键驱动编写(中断方式)
在Linux驱动之按键驱动编写(查询方式)已经写了一个查询方式的按键驱动,但是查询方式太占用CPU,接下来利用中断方式编写一个驱动程序,使得CPU占有率降低,在按键空闲时调用read系统调用的进程可以 ...
- Git 在团队中的使用--如何正确使用Git Flow
Git的优点 Git的优点很多,但是这里只列出我认为非常突出的几点. 由于是分布式,所有本地库包含了远程库的所有内容. 优秀的分支模型,打分支以及合并分支,机器方便. 快速,在这个时间就是金钱的时代, ...