开始之前,有几点要理解的:
1、当程序运行的时侯,系统默认开启了一个主线程(假设命名mainThread)。通常我们在Main()函数中定义主线程的工作。
2、可以在Main()函数中启动多个子线程:Thread-1,Thread-2,Thread-3等。
3、主线程有可能比子线程先结束运行。
第三点是关键,这也是要用到join()方法的一个重要原因,因为有时侯我们希望主线程等待子线程执行完成之后再结束。
看一个例子:在主线程中开启一个子线程,用来计算1至100的和,然后在主线程中打印出来
package JConcurrence.Study;
public class JoinDemo {
static int result = 0;
public static void main(String[] args) {
Thread subThread = new Thread() {
@Override
public void run() {
for (int i = 1; i <= 100; i++) {
result = result + i;
/*模拟耗时操作*/
try {
Thread.sleep(20);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
};
subThread.start();
System.out.print(result);
}
}
程序运行结果:
1到100求和结果等于:0
出错了。程序运行进入Main()函数后,开启子线程subThread计算求和。此时主线程并没有停止,继续往下运行。子线程subThread运行耗时大约2秒,而主线程如出膛子弹迅速往下执行完毕。子线程此该还没有反应过来,主线程已经输出了结果。
为了输出正确的结果,显而易见,必须让主线程等待子线程运行完毕再执行System.out.print。
这时,轮到Thread.Join()出场了。
在subThread.start()和System.out.print(result)之间加上
try {
subThread.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
程序运行结果:
1到100求和结果等于:5050
结果正确!
Join()方法中可以设置值,即等待多久。比如上面如果把subThread.join()改为subThread.join(1000),就是告诉主线程等待子线程1秒钟后再继续运行。你可以这样修改后试着运行一下程序,这时程序输出的应该是0到5050间的一个值。
注意事项
在多个子线程的情况下,若依次执行每个线程的start()和join()方法,则各个线程之间是同步的。
for(int i=1;i<=3;i++){
Thread subThread = new Thread(new workrun(i));
subThread.start();
subThread.join();
}
上面程序中的三个子线程会顺序同步的执行。
要想三个子线程能够并发执行,需要改变join调用位置
Thread subThread1 = new Thread(new workrun(1));
Thread subThread2 = new Thread(new workrun(2));
Thread subThread3 = new Thread(new workrun(3));
subThread1.start();
subThread2.start();
subThread3.start();
subThread1.join();
subThread2.join();
subThread3.join();