join()方法的使用

1.join()方法介绍

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

join(long):等待一定时间。

  • 在join过程中,如果当前线程被中断,则当前线程出现异常,但join所属的线程继续运行。
  • join()在内部使用wait()方法进行等待,所以会释放锁
package chapter3.join;

public class MyThread extends Thread{
	
	@Override
	public void run() {
		super.run();
		try {
			int secondValue = (int) (Math.random()*10000);
			System.out.println(secondValue);
			Thread.sleep(secondValue);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

}

package chapter3.join;

public class Test {

    public static void main(String[] args) {
        try {
            MyThread myThread = new MyThread();
            myThread.start();
            myThread.join();
            System.out.println("MyThread 执行完之后执行!");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

运行结果:

9574
MyThread 执行完之后执行!

2.join()方法后面的代码提前运行

join()方法源码:

//不带参数
public final void join() throws InterruptedException {
     join(0);
}
//带参数
public final synchronized void join(long millis) throws InterruptedException { long base = System.currentTimeMillis(); long now = 0; if (millis < 0) { throw new IllegalArgumentException("timeout value is negative"); } if (millis == 0) { while (isAlive()) { wait(0); } } else { while (isAlive()) { long delay = millis - now; if (delay <= 0) { break; } wait(delay); now = System.currentTimeMillis() - base; } } }
package chapter3.join;

public class ThreadA extends Thread{
	
	private ThreadB threadB;
	
	public ThreadA(ThreadB threadB) {
		this.threadB = threadB;
	}
	
	@Override
	public void run() {
		super.run();
		try {
			synchronized (threadB) {
				System.out.println("begin A ThreadName="+Thread.currentThread().getName()+"--"+System.currentTimeMillis());
				Thread.sleep(3000);
				System.out.println("end A ThreadName="+Thread.currentThread().getName()+"--"+System.currentTimeMillis());
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

}

package chapter3.join;

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

}

package chapter3.join;

public class Test {

	public static void main(String[] args) {
		try {
			ThreadB threadB = new ThreadB();
			ThreadA threadA = new ThreadA(threadB);
			threadA.start();
			threadB.start();
			threadB.join(2000);
			System.out.println("main end:"+System.currentTimeMillis());
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
}

 运行结果1:

begin A ThreadName=Thread-1--1561019085673
end A ThreadName=Thread-1--1561019088673
main end:1561019088673
begin B ThreadName=Thread-0--1561019088673
end B ThreadName=Thread-0--1561019091673

  • threadB.join(2000)先抢到threadB锁,然后释放;
  • ThreadA抢到锁,打印begin,sleep;
  • 打印end,释放锁;
  • threadB.join(2000)和ThreadB争抢锁,join再次抢到,发现时间已过,释放然后打印main end;
  • ThreadB抢到锁,打印begin,end;

运行结果2:

begin A ThreadName=Thread-1--1561019572226
end A ThreadName=Thread-1--1561019575226
begin B ThreadName=Thread-0--1561019575226
end B ThreadName=Thread-0--1561019578226
main end:1561019578226

  • threadB.join(2000)先抢到threadB锁,然后释放;
  • ThreadA抢到锁,打印begin,sleep;
  • 打印end,释放锁;
  • threadB.join(2000)和ThreadB争抢锁,ThreadB抢到,打印begin,sleep,打印end后释放锁;
  • mian end最后输出;
原文地址:https://www.cnblogs.com/hunter-56213/p/11059931.html