本帖最后由 舒远 于 2012-9-9 01:30 编辑
采用继承Thread类的方式来创建多线程,程序会有一个潜在的危险。
public class InvokeRun extends Thread{
private int i;
//重写run方法
public void run(){
for(;i < 100; i++){
//直接调用run方法时,Thread的this.getName返回该对象名字
//而不是当前线程的名字
//使用Thread.currentThread().getName()总是获取当前线程的名字
System.out.println(Thread.currentThread().getName()+" " + i);
}
}
public static void main(String[] args){
for(int i = 0; i < 100; i++){
System.out.println(Thread.currentThread().getName()) + " "+i)
if(i == 20){
//直接调用线程对象的run方法
//系统会把线程对象当成普通对象,把run方法当成普通方法
//所以,下面两行代码并不会启动2条线程,而是依次执行2个run方法.
new InvokeRun().run();//a
new InvokeRun().run();//b
}
}
}
}
上面程序试图在主线程中 i == 20时创建并启动2条线程,编译该程序,一切正常,运行该程序,
发现改程序只有一条线程---main线程。程序的执行过程如下:
1输出main 20之后。又重新开始输出main 0。
2从main 0一直输出到main 99.再次从main 0开始输出。
3再次从main 0一直输出到main 99。再次从main 22开始输出。知道main99结束
上面程序始终只有一条线程,并没有启动任何新线程。关键是因为a和b处调用了线程对象的run方法。
而不是start方法。启动线程应该调用start方法而不是run方法。
如果程序从未调用线程对象的start方法来启动它,那么这个线程对象将一直处于“新建”状态。
它拥有也不会作为线程获得执行的机会,它只是一个普通的java对象。当程序调用线程的run方法时,
与调用普通java对象的普通方法并无任何区别。因此绝对不会启动一条新线程。 |