Java多线程/并发07、Thread.Join()让调用线程等待子线程

来源:互联网 发布:紫金桥软件 编辑:程序博客网 时间:2024/05/21 09:26

开始之前,有几点要理解的:
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();
0 0