这篇文章需要认认真真仔仔细细的看才能看懂
一. 普通对象与函数对象 JavaScript 中,万物皆对象!但对象也是有区别的。分为普通对象和函数对象,Object ,Function 是JS自带的函数对象。下面举例说明 function f1(){}; var o3 = new f1(); console.log(typeof Object); //function |
|
在上面的例子中 o1 o2 o3 为普通对象,f1 f2 f3 为函数对象。怎么区分,其实很简单,凡是通过 new Function() 创建的对象都是函数对象,其他的都是普通对象。f1,f2,归根结底都是通过 new Function()的方式进行创建的。Function Object 也都是通过 New Function()创建的。 二. 原型对象 注:普通对象没有prototype,但有__proto__属性。 原型对象[xxx.prototype]其实就是普通对象(Function.prototype除外,[Function.prototype]是函数对象,但它很特殊,他没有prototype属性(前面说函数对象都有prototype属性))。看下面的例子: 从这句console.log(f1.prototype) //f1 {} 的输出就结果可以看出,f1.prototype就是f1的一个实例对象。就是在f1创建的时候,创建了一个它的实例对象并赋值给它的prototype,基本过程如下: 所以,Function.prototype为什么是函数对象就迎刃而解了,上文提到凡是new Function ()产生的对象都是函数对象,所以temp1是函数对象。 那原型对象是用来做什么的呢?主要作用是用于继承。举了例子: 从这个例子可以看出,通过给person.prototype设置了一个函数对象的属性,那由person实例(例中:zjh)出来的普通对象就继承了这个属性。具体是怎么实现的继承,就要讲到下面的原型链了。 三.原型链 console.log(zjh.__proto__ === person.prototype) //true 同样,person.prototype对象也有__proto__属性,它指向创建它的函数对象(Object)的prototype console.log(person.prototype.__proto__ === Object.prototype) //true 继续,Object.prototype对象也有__proto__属性,但它比较特殊,为null(记住Object是函数对象) console.log(Object.prototype.__proto__) //null 我们把这个有__proto__串起来的直到Object.prototype.__proto__为null的链叫做原型链。如下图: 画图约定: 疑点解释: 2.Function.__proto__ === Function.prototype // true typeof Object =》function typeof Object.prototype =>object typeof Function =》function typeof Function.prototype =>function 自己是由自己创建的,好像不符合逻辑,但仔细想想,现实世界也有些类似,你是怎么来的,你妈生的,你妈怎么来的,你姥姥生的,……类人猿进化来的,那类人猿从哪来,一直追溯下去……,就是无,(NULL生万物) 3.Function.prototype.__proto__ === Object.prototype //true 五.constructor 完善下上面的内存结构图: 有两点需要注意: 六.总结 要深入理解这句话,我们再举个例子,看看前面你真的理解了吗? animal.price = 2000;// console.log(dog.price) //undefined 为什么呢?画一下内存图: 这说明什么问题呢,执行dog.price的时候,发现没有price这个属性,虽然prototype指向的animal有这个属性,但它并没有去沿着这个“链”去寻找。同样,执行tidy.price的时候,也没有这个属性,但是__proto__指向了animal,它会沿着这个链去寻找,animal中有price属性,所以tidy.price输出2000。由此得出,原型链的真正形成是靠的__proro__,而不是prototype。 最后打个比喻,虽然不是很确切,但可能对原型的理解有些帮助。 |