1.概述prototype与__proto__
之前就了解过一些原型与原型链的问题,觉得十分抽象,知道__proto__与prototype是什么。__proto__是所有对象里面都有的一个属性,他通常是一个对象(有一个特殊情况就是Object.prototype.__proto__===null)。而prototype则是函数里都有的属性,他通常是一个对象。
对象.__proto__通常存放对象的公共属性。函数.prototype通常存放函数的公共属性。因为函数也是对象,因此函数通常会既有__proto__属性,也有prototype属性。特殊情况如Function.__proto__===Function.prototype。
2.什么是原型?
原型其实就是共有属性!我们把这些共有属性放到一个对象中,通过引用的方式,使所有拥有原型的对象都能够使用其中的公共属性。这个存放共有属性的对象就是原型,而 '对象.__proto__'与 '函数.prototype'都是其对应原型的引用,可以查看对象或者函数的公共属性,方法。
3.原型产生的原因
我们知道数据是在内存中存储的,占据着内存。而很多属性是通用的,比如说toString(),valueOf()等等,如果在每个对象里面都存放这些属性,那么随着内存中对象的增多,会占用巨大的空间,造成内存空间的浪费。
而原型解决了这一问题,他将公共属性存放在一个对象之中,对公共属性通过引用的方式,连接起来,这样,即使对象再多,他们使用的都是同一个地址中的公共属性,即使用了原型,不会重复的浪费内存。
4.不同的原型
原型也不仅仅只有一个。例如
1.一般对象的原型
上图是表示一般对象的__proto__,他存放了Object的原型。即也是Object.prototype。
上图很容易看出,obj.__proto__===Object.prototype,他们都是公共属性对象的引用。
2.Number对象的原型
上图可见,用Number构造函数实例化的num对象num。而num.__proto__与Object.prototype不相等。即num对象的原型不是一般对象的原型。
上图可见,num.__proto.__proto__才是一般对象的原型。这个也就形成了原型链。
通过Number构造函数实例化出来的对象num的原型是在一般对象原型上一层,这意味着,num对象优先在num.__proto__原型里面找公共属性,如找不到,再到num.__proto__.__proto__原型里面找公共属性。
3.其他原型类似于Number对象的原型。
5.原型链
我们知道了,不同构造函数的对象,原型是不同的,而他们的根原型都是一般对象的原型,即Object.prototype。通过4.2中对Number对象原型中的描述,可以看出其与一般对象原型的关系。
因此我们可以得到下图:
不同的名字代表了不同的构造函数实例出来的不同类型的对象。通过这样一级一级在原型中查找公共属性的方式,就构成了原型链。
6一个公式
var 对象=new 函数()
对象.__proto__===函数.prototype
7.补充一点函数的__proto__与prototype
我写了一串实验了一下
我实验的这些构造函数,他们都__proto__都是Function.prototype。说明了,这些构造函数,包括Function函数自身,都是由Function构造函数创建的,因此他们的公共属性原型都等于Function.prototype。
需要注意一点的是,构造函数都prototype属性对应的原型,是该构造函数实例对象的__proto__属性对应的原型。而构造函数作为对象的__proto__都是Function.prototype。