天行健、君子以自强不息;地势坤、君子以厚德载物!

关于我

zhenn 前端工程师,淘宝 北京!这里仅仅记录我的技术生活以及成长历程,如果有兴趣和我交流,猛击以下链接即可。
follow zhenn in TC Microblog
follow zhenn in SinaTwitter

你真的了解prototype吗?

JavaScript中的prototype,即构造函数的原型对象,我想对于使用过JavaScript oo模式编程的你来说,肯定都能很好的利用prototype的特性来编写出优美的面向对象的代码,因为它的确很好用。但是你真的足够了解prototype吗?

首先我觉得,在分析prototype之前,应该先了解为什么要使用prototype来实现继承,而并不是把所有属性写到构造函数内部,那么这样做优势体现在哪里呢,如果你跟我有一样的困惑,请看如下两段代码:

/*代码1*/
var F = function(){
	this.a = 1;
	this.b = 2;
	this.c = 3;
	this.d = 4;
	this.e = 5;
	this.f = 6;
};
for(var i=0;i<100000;i++){
	eval('var f' + i + '= new F;');
}
/*代码2*/
var F = function(){};
F.prototype = {
	a: 1,
	b: 2,
	c: 3,
	d: 4,
	e: 5,
	f: 6
};
for(var i=0;i<100000;i++){
	eval('var f' + i + '= new F;');
}

在浏览器中分别执行两段代码,观察浏览器的内存占用情况(数据量比较大,等待脚本执行完毕,浏览器稳定后的数据),如下图:

从图片中的数据不难看出,在Firefox中,把所有属性全部写到构造的原型对象中,生成100000个实例对象时,占用内存会比把所有属性全部写 到构造内部少15M,而在IE中,这个差距竟然达到97M,匪夷所思啊,神奇的IE。这也就是为什么写oo代码时,总是喜欢把类的一些属性和方法写到它的 原型对象中,在达到同样目的的前提下节省内存的使用率,我想没有哪一个人愿意拒绝这样的做法。

缘何使用prototype会节省内存空间呢?我觉得有必要了解JavaScript中类的实例化过程。

var F = function(){
	this.a = 1;	
};
F.prototype = {
	a: 2	
};
var f = new F;
	
/*new F的过程*/
var f = {};    //创建空对象
f.__proto__ = F.prototype;  //设置对原型对象的引用
F.call(f);     //拷贝构造中的属性,初始化对象

通过这个实例化过程,可以看到,在每个实例化对象中,如果构造中属性全部放到原型对象中来实现继承(所谓的原型链继承),并不是拷贝原型中的全部属 性到自身所占的空间内,而是通过__proto__来实现对F.prototype的引用,显而易见,这样做更能节省内存空间。

事实上,这个过程可以解释我们再开发中遇到的很多问题,比如,在构造内部定义的属性要比在原型中定义的属性优先级要高,正如上述代码片段,f.a =1而不是f.a=2,就是因为对原型对象的引用发生在对象初始化之前,详细见下图:

另外一个问题,修改原型对象中的属性或方法,即使在实例化之后,依然会奏效,这里需要注意的是,这个修改的过程需在F.prototype指向的原始对象上修改,不要试图将prototype重新指向一个新的对象,因为在实例化之后,在实例对象f中查找某个属性时,依然会从F.prototype指向的原始对象中查找,也就是说并不没有改变f.__proto__的指向,虽然我们期望看到他重新指向一个新的对象。

PS:说了很多,重点只有一个,就是深刻类的实例化过程,这才是JavaScript原型继承的本质。

posted on 2011-02-20 18:46  zhenn  阅读(879)  评论(0编辑  收藏  举报

导航