Java多线程-join的使用

join的作用

join的作用是等待线程对象销毁。

方法join 的作用是使所属的线程对象 x 正常执行 run() 方法中的任务,而使当前线程 z 进行无限期的阻塞,等待线程 x 销毁之后再继续执行当前线程 z 后面的代码。

注意:

如果方法 join() 与 interrupt() 方法相遇,则会抛出异常。示例如下:

public class JoinException {
    public static void main(String[] args) {
        try {
            ThreadB b = new ThreadB();
            b.start();
            Thread.sleep(500);
            ThreadC c = new ThreadC(b);
            c.start();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    static class ThreadA extends Thread {
        @Override
        public void run() {
            for (int i = 0; i < Integer.MAX_VALUE; i++) {
                String newString = new String();
                Math.random();
            }
        }
    }

    static class ThreadB extends Thread {
        @Override
        public void run() {
            try {
                Thread a = new ThreadA();
                a.start();
                a.join();
                System.out.println("线程B在 run end 处打印了");
            } catch (InterruptedException e) {
                System.out.println("线程B在 catch 处打印了");
                e.printStackTrace();
            }
        }
    }

    static class ThreadC extends Thread {
        private ThreadB threadB;

        public ThreadC(ThreadB threadB) {
            this.threadB = threadB;
        }

        @Override
        public void run() {
            threadB.interrupt();
        }
    }
}

运行结果如下:

 

 

join()会释放锁

Thread.sleep()不释放锁。而 join() 方法会释放锁

方法 join(long) 的功能在内部是使用 wait(long) 方法来实现的,所以 join(long) 方法具有释放锁的特点。

方法 join(long) 源代码如下:

 

方法 join() 后面代码提前运行:出现意外

join(long millis) 设置了时间参数的 join() 容易出现意外。示例如下:

tips:最后的解决方案是把 b.join(2000) 替换成 b.join()

public class JoinTest {
    public static void main(String[] args) {
        try {
            ThreadB b = new ThreadB();
            ThreadA a = new ThreadA(b);
            a.start();
            b.start();
            b.join(2000);
            System.out.println("    main edn " + System.currentTimeMillis());
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    static class ThreadA extends Thread {
        private ThreadB b;

        public ThreadA(ThreadB b) {
            this.b = b;
        }

        @Override
        public void run() {
            synchronized (b) {
                try {
                    System.out.println("begin A ThreadName=" + Thread.currentThread().getName() + " " + System.currentTimeMillis());
                    Thread.sleep(5000);
                    System.out.println("    end A ThreadName=" + Thread.currentThread().getName() + " " + System.currentTimeMillis());
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    static class ThreadB extends Thread {
        @Override
        synchronized public void run() {
            try {
                System.out.println("begin B ThreadName=" + Thread.currentThread().getName() + " " + System.currentTimeMillis());
                Thread.sleep(5000);
                System.out.println("    end B ThreadName=" + Thread.currentThread().getName() + " " + System.currentTimeMillis());
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

运行结果分析:

 

 

 

 

posted @ 2020-03-24 01:02  lkc9  阅读(382)  评论(0编辑  收藏  举报