2 brucecloud brucecloud 于 2015.07.13 14:08 提问

Java 多线程问题求解....

public class SigleThread extends Thread {

@Override
public void run() {
    for (int i = 0; i < 5; i++) {
        System.out.println(i);
        try {
            join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

}

public class Test {
public static void main(String[] args) {
SigleThread sing = new SigleThread();
sing.start();
}
}

这个时候控制台在输出0之后....就会处于卡死状态....为什么会这样????求大神解惑~~~

而换成下面这种调用方式就没问题....

public class Test {
public static void main(String[] args) {
Runnable sing = new SigleThread();
Thread t = new Thread(sing);
t.start();
}
}

这种调用就会打印0 1 2 3 4 5然后正常结束....请问到底是什么原理??

2个回答

lingfeiwen
lingfeiwen   2015.07.13 16:14
已采纳

主要的原因是楼主没搞清楚,run方法里面的this指的是谁。

看我贴的代码,楼主拿去运行下this是谁。语句后面我写了注释,楼主注意看下。

 public class SigleThread extends Thread{
    @Override
    public void run() {

        System.out.println(this); //this是 sing对象

        System.out.println(this.isAlive()); //sing是个线程,但是此线程并没有start,它不是活的

        for (int i = 0; i < 5; i++) {
            System.out.println(i);
            try {
                this.join(); //关键是在这里。join时检测this是否是活的(可以去看下join的源码),也就是检测isAlive是不是返回true。
                            //如果是,则线程等待,即wait(0),在没有被notify之前,就阻塞在那里。也就是楼主说的卡死在那里了。
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
 public class Test {
    public static void main(String[] args){

        Runnable sing = new SigleThread();//new了一个线程。叫Thread-0
        System.out.println(sing);
        Thread t = new Thread(sing);//又new了一个线程。叫Thread-1。Thread-1启动后,run方法里面的this是指sing,而不是t自己,执行的是sing的run方法。
        System.out.println(t);
        t.start(); //Thread-1启动。但Thread-0并没有启动。

    }
}
brucecloud
brucecloud 回复菩萨心肠: 你说完全正确~~~昨天晚上我看了JVM底层源代码~~~只有在调用start()方法的时候才会让其处于活着的状态,也就是isAlive()是true,而且在join()的时候会去检查线程是不是活的,不是活的join()是无任何作用的~~你的代码写的很清楚,原理讲的也很清楚~~赞一个~~~谢啦~~
2 年多之前 回复
lingfeiwen
lingfeiwen 如果是SigleThread sing = new SigleThread(); sing.start();去运行的话,run里面的this就是知sing线程自己,sing.start();后sing就是活的了。既然是活的,在join的时候就会wait,直到有其他线程来唤醒它,才能醒过来执行,显然在例子中没有其他线程来执行唤醒动作,所以就像楼主说的那样,卡死在那里了。楼主现在能明白吗?
2 年多之前 回复
woyaochenggong774
woyaochenggong774   2015.07.13 15:08

join()后面的代码只能等待到子线程任务完成之后去执行,而你的join()放在了子线程的循环之中,只有第一次是执行了,而后面的循环都属于join()后面的代码,你怎么让他执行。而第二种你用的方法强转成Runnable(), join()是Thread的方法,不是Runnable的方法。个人理解。

brucecloud
brucecloud 没错,我也是这么理解的...但是就是没找到认证这种理解对不对的有利依据.....
2 年多之前 回复
Csdn user default icon
上传中...
上传图片
插入图片