Thread.join()
为什么要用join()方法
如果子线程里要进行大量的耗时的运算,主线程往往将于子线程之前结束,但是如果主线程处理完其他的事务后,需要用到子线程的处理结果,也就是 主线程需要等待子线程执行完成之后再结束,这个时候就要用到join()
方法了
join() 的作用
让父线程等待子线程结束之后才能继续运行
join() 一共有三个重载版本,分别是无参、一个参数、两个参数:
实列:
public class TestJoin {
public static void main(String[] args) throws InterruptedException {
// TODO Auto-generated method stub
ThreadTest t1=new ThreadTest("A");
ThreadTest t2=new ThreadTest("B");
t1.start();
t2.start();
}
}
class ThreadTest extends Thread {
private String name;
public ThreadTest(String name) {
this.name = name;
}
public void run() {
for (int i = 1; i <= 5; i++) {
System.out.println(name + "-" + i);
}
}
}
结果: 可能无 t1 线程没有执行完毕 就执行t2
加上 join() 在主线程中调用t.Join(),也就是在主线程中加入了t线程的代码,必须让t线程执行完毕之后,主线程(调用方)才能正常执行
public class JoinTest {
public static void main(String[] arg) {
ThreadTests t = new ThreadTests("Li");
ThreadTests t2=new ThreadTests("cheng");
t.start();
System.out.println("t");
try {
t.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("t2");
t2.start();
}
}
class ThreadTests extends Thread {
private String name;
public ThreadTests(String name){
this.name=name;
}
public void run() {
for (int i = 1; i <= 2; i++) {
System.out.println("name:"+name+"i="+ i);
}
}
}
结果:
t
name:Lii=1
name:Lii=2
t2
name:chengi=1
name:chengi=2
其中
(1) 三个方法都被final修饰,无法被子类重写。
(2) join(long), join(long, long) 是synchronized method,同步的对象是当前线程实例。
(2) 无参版本和两个参数版本最终都调用了一个参数的版本。
(3) join() 和 join(0) 是等价的,表示一直等下去;join(非0)表示等待一段时间。
从源码可以看到 join(0) 调用了Object.wait(0),其中Object.wait(0) 会一直等待,直到被notify/中断才返回。
while(isAlive())是为了防止子线程伪唤醒(spurious wakeup),只要子线程没有TERMINATED的,父线程就需要继续等下去。
(4) join() 和 sleep() 一样,可以被中断(被中断时,会抛出 InterrupptedException 异常);不同的是,join() 内部调用了 wait(),会出让锁,而 sleep() 会一直保持锁。
posted on 2020-08-14 10:51 shumeigang 阅读(207) 评论(0) 收藏 举报