作者:qlampskyface
和作者联系:xiaozuidaizhi@sina.com
众所周知,this的两种用法非常普遍.
1.代替当前实例.例如this.成员变量名,this.成员方法名
2.代替构造函数,比如this().
上述两点不再赘述.
但是,我觉得this还有一种现象大家可能没有注意到:
可能有时候你会发现如下的现象:
public class Test implements Runnable{
Thread thread1;
Thread thread2;
public Test(){
thread1 = new Thread(this,"1");//---------------------->|
thread2 = new Thread(this,"2");//---------------------->|
}
public static void main(String args[]){
Test t = new Test();
t.startThreads();
}
public void run(){
//do thread's things
}
public void startThreads(){
thread1.start();
thread2.start();
}
}
在|处,你可能很奇怪:为什么可以在还没有创建的时候就可以自己调用自己呢?
为了讲清楚上面的现象,必须先讲清楚构造函数的继承.
如果有两个类成继承关系,那么
如果缺省构造器不显式声明,那么子类构造器必先调用其父类缺省构造器(其实我觉得叫第一构造器更为合适,就是没有入参的构造器),再调用自己的构造器。
下面我们接着谈上面奇怪的现象,为什么还可以在没有创建子类对象的时候就可以用this代替本身呢?那是因为在调用子类构造器之前,已经调用了父类的缺省构造器(父类为Object).你还可以让Test继承一个类,看看是否先调用其父类的构造器.
在上文中,由于自身能力有限,并没有将这个话题深入讨论,只是肤浅得作以解释,希望能够抛砖引玉.请指正!
更正:由于本人才疏学浅,对上面现象的解释发生了谬误,在此向大家致以诚挚的歉意。多亏DreamHead兄看到在下这篇拙文,提出了宝贵的意见,才能够去伪存真,下面,我将上面之现象的真正原因补充如下:
从C/C++过来的人可能会对这个过程有比较深刻的认识。 以C为例,通常我们动态分配的工作是这样完成。
1 调用malloc分配一块内存空间,但这块内存是只是一块拥有对象所需大小的内存,并没有初始化。
2 调用相应初始化方法对这块内存进行初始化。 到了C++中,new其实就是这两个方法的和,我们通常用的时候只看到new在分配内存,实际上,它还会调用构造函数(相当于刚才的初始化方法)完成初始化。
Java的过程也是一样。在Java里除了基本类型都是对象,我们持有的都是引用(功能上类似于C语言中的指针,但绝非指针!)。对象的构造过程分为分配内存和调用构造函数两个步骤。在程序执行到构造函数时,分配内存的动作已经完成。上面的程序中,Thread构造函数传入的是一个引用,也就是一个地址,而此时因为Test已经分配了内存,所以把它的地址作为参数传入当然OK了。 至于本人所说的基类构造函数的问题,没说错,只是和这个问题不相干,请大家注意。
再次谢谢DreamHead兄。