关于javascript实现类的本质的探讨

不少人对javascript实现类这一点上有些疑惑。因为本来一个很普通的function怎么会变成了一个类的构造函数。在这里我想根据我这几天的研究。谈谈javascript是怎么样实现class的。
讲解这个话题之前我们必须先了解一些背景知识。
首先我们必须了解一点在javascript中。对象是使用类似于Dictionary的机制来实现.请看下面代码 
var  Obj  =   new  Object();
Obj[
" Age " =   32 ;
Obj.Name 
=   " Hello " ;
alert(
" Age:  " +  Obj.Age  +   " Name: " +  Obj[ " Name " ]);


所以我们可以看出实际上在javascript中就每一个对象类似于Dictonary。对该对象的属性进行访问。也就是从该Dictonary中读取也就是说Obj.Age实际上也就是同Obj["Age"]是同样意义的。

然后我们需要理解借用函数这一个概念。还是先看看代码

function  Speak()
{
    alert(
"Age: "+ this.Age+ "Name:"+ this.Name);
}




var  Obj  =   new  Object();
Obj.Age 
=   32 ;
Obj.Name 
=   " Hello " ;

Speak.call(Obj);

在调用了函数Speak的call之后。函数中的全部的this都会变成call的第一个参数。而如果代码如下:
function  Speak()
{
    alert(
"Age: "+ this.Age+ "Name:"+ this.Name);
}




var  Obj  =   new  Object();
Obj.Age 
=   32 ;
Obj.Say 
=  Speak;

Obj.Say();

这个时候Speak函数中的this也会变成Obj这同Seapk.call(Obj)是等价的


了解了这两点后。我们就可以看看new之后发生了一些什么.

function  ClassA()
{
    
this.Name = "Class A";
}

ClassA.prototype.Speak 
=   function ()
{
    alert(
this.Name);
}

    
    
function  SimulateNew(constructor)
{
    
var obj = new Object();
    
    
for(var s in constructor.prototype)//1
    {
        obj[s] 
= constructor.prototype[s];
    }


    constructor.call(obj);
//2
    //1,2的次序在某些情况下又影响.new应该是按照先1后2来进行的
    return obj;
}


var  a  =   new  ClassA();
a.Speak();

var  anotherA  =  SimulateNew(ClassA);
anotherA.Speak();
所以我们可以看出从本质上来所.new在幕后做得.也就是和SimulateNew函数做得是类似的(当然还要处理一些注册等的工作用来处理instanceof等reflection).还值得一提的是在每一个函数被声称后都会自动加上一个prototype的属性
function  NoramlFunction()
{
    
//do anything...    
}


alert(NormalFunction.prototype);

可以看到prototype是自动添加上的.相当于:
function  NoramlFunction()
{
    
//do anything...    
}

// NormalFunction.prototype = new Object; 这一步在生成function对象的时候已经由后台自动实现了
alert(NormalFunction.prototype);


理解了这一点后.我们可以很好的理解javascript的继承等等概念
关于javascript的继承有兴趣的可以看一下的帖子:
http://birdshome.cnblogs.com/archive/2005/01/28/95933.html
关于javascript面向对象更详细的解释
http://mckoss.com/jscript/object.htm

注:第二个链接的私有成员的成立本人持保留意见.这个是一个很tricky的办法.而且在prototype定义的函数中无法访问私有成员而且私有成员不能通过this访问.这违背了private的本质
所以应该按照Professional.JavaScript.For.Web.Developers一书88页(英文原版Apr.2005)中说的使用命名来区分用途

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值