前几天当我同时向往说起node.js的时候,我逐渐发现原来JS是个多么丰富语言——我原来一直都在井底(初学者的悲哀)。当我想认认真真学习JavaScript的时候,我发现prototype与__proto__、constructor这几个家伙老是让我很有挫败感(学一门语言属性都搞不懂,还学个P)。但我是一个不轻易服输的人,也不是不会妥协,就是有这种习惯,我就要去做就要去弄懂哪怕挫折感更多。于是,我在网上搜大神们的对这几个让我烦恼的东西的阐述与解释。最后的结果是--还是搞不懂。“什么对象原型,什么内部原型”,难道是孙悟空吗?这么多变身。后来在看“javascript语言精髓与编程实践精简版”的时候突然让我了解了__proto__与prototype的真实面目。
这里我借鉴一张图([url]http://anykoro.sinaapp.com/2012/01/31/javascript%E4%B8%ADfunctionobjectprototypes__proto__等概念详解[/url])其他的文章如果想真正了解的就不要看了,越看越迷糊(有些人不懂或者说半懂就给上面的三个家伙乱下定义,搞的我云里雾里,误人子弟)。我在这里只用比喻,也就说我说的prototype与__proto__与constructor是我所理解的模型,而不是给他死下定义,每个人的理解是可以不同的知识不要搞的那么专业化,像是你能给他下定义似得(装大神,误人子弟)。
function能够调用的prototype一般被定义为“对象原型的引用”,有的人也说是"实例"(一开始我以Java的理念理解,真是把我搞醉了)。我对于prototype的理解就是“爸爸”,也就是对象的模子。当我们对prototype添加属性的时候"模子"会改变它的形状,以后用这个模子造出来的实例(var x = new X;)就会是按照模子来做的,所以说他的实例就会拥有prototype的属性。如果我们用"X.prototype="的形式就会改变它的模子也就是换一个“爸爸”,所以这样做不会是一点点改变了,而是替换(就是找继父)。所以说在执行这句话的时候,就会改变X的生出来的孩子的属性。如果还是不懂就弄清我给你们的链接加上自己做实验。
__proto__在我的理解里是“影子”或者说父类的属性视图。用 "var X = function (){}或者function X() {} var x = new X();"的时候x中的__proto__属性会指向X.prototype.这是原型链中最恒等的,不管prototype改变还是__proto__改变,这条链始终是要相等。
<code>
//原文的例子,有些许错误。在他的例子上有些许改进。
function Foo(){ } Foo.__proto__.test="__proto__ test property found";//通过__proto__扩展Foo.__proto__.addextend=function(){alert("Foo add extend by __proto__");}
alert(Function.prototype.test);//可以访问,这里为什么对象的是Function.prototype呢?相信Foo.__proto__是儿子,而爸爸是谁呢?Function.prototype
var foo=new Foo;
foo.__proto__.lab = "abc";
alert(foo.__proto__===Foo.prototype);
</code>
上面这段代码对Foo的__proto__进行添加(其实对Foo.__proto__的改变按道理说是对应Function.prototype.test,foo的__proto__是对应Foo的prototype,这条关系永远的恒等的),而__proto__是特殊的影子或者视图(就像Oracle数据库),你修改它是会影响到Function的prototype的,如果你在上面代码后面添加
<code>
for(var x in foo.__proto__){ alert(x);}for(var y in Foo.prototype){ alert(Foo.prototype.lab);}
alert(foo.lab);
</code>
是可以访问到lab(中文翻译:实验)这个属性的。所以说,就这一单来说__proto__类似于数据库中的视图,这是很奇特的就像儿子反过来会影响爸爸(多好的一家人)。
而理解constructor的时候可以把prototype看做是一个实例,就如Foo的prototype(实例)的构造函数不就是指向函数本身吗!也就是说Foo.prototype.constructor的与foo.constructor的是一样的。而Foo.constructor是Function(这里可以解释为构造Foo的构造函数不就是new Function())这里其实对于prototype的理解我也不能很好的去统一,大家只能自己意会了(分开理解会更加清楚)。
总之,我认为(以我学习几天的JavaScript的立场)来说,Foo.prototype就是就是其new 出来的实例的__proto__的 "爸爸"或者说"真身",而foo.__proto__的则是类的“影子”或者是“视图”(特殊的),当爸爸改变的时候生出来的儿子当然不同,但是儿子改变是爸爸也会变化(只能自己理解,我真说不清楚了)。而constructor就是构造对象的属性了,注意是对象(JavaScript中一切皆对象,所以function本身也是对象,也会有constructor的,所有产生function的都是大Function(通过 new Function();的方式可以看到)).小foo的构造当然是 大Foo,大Foo的实例(prototype)的构造函数当然也是大Foo.所以才会有Foo的prototype的constructor指向他本身的说法。完毕!(PS:虽然是刚刚学习JavaScript,但我认为这样理解这三个东西,是我唯一能够接受的理解方式了,有不同的理解又不产生歧义的可以用自己理解的方式。如果,你看完任然感觉迷惑,就按照你自己的来吧)