java线程join方法

1.join方法

join方法的作用是进行线程插队,也就是说调用了join方法的线程相对于调用它的上级线程拥有跟高的执行权。调用join方法的线程的上级线程必须等待调用join方法的线程执行完成才能继续执行。

举个小例子:

public class testjoin{

	public static void main(String[] args) throws InterruptedException {
		// TODO Auto-generated method stub
		Thread t1 = new Thread(new t1(),"t1");
		t1.start();
		
		
	
	}			

}

class t1 implements Runnable {

	@Override
	public void run() {
		// TODO Auto-generated method stub
		for (int i = 0; i < 10; i++) {
			System.out.println(Thread.currentThread().getName()+"->"+i);
			if(i==3) {
				Thread t2 = new Thread(new t2(),"t2");
				t2.start();
				try {
					t2.join();    //调用join方法,线程t1必须等待t2执行完毕才能继续执行。  
				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
		}
		
	}
}

class t2 implements Runnable {

	@Override
	public void run() {
		// TODO Auto-generated method stub
		for (int i = 0; i < 10; i++) 
			System.out.println(Thread.currentThread().getName()+"->"+i);
			
		
		
	}
}

运行结果如下:

t1->0
t1->1
t1->2
t1->3
t2->0
t2->1
t2->2
t2->3
t2->4
t2->5
t2->6
t2->7
t2->8
t2->9
t1->4
t1->5
t1->6
t1->7
t1->8
t1->9

无论运行多少次,运行结果都不变,原因是t2线程运行了join方法,那么它的上级线程t1就必须等待t2运行完后才能继续调用。

2.join源码

join方法能传入一个常数,millis是要主线程暂停时间。

1.如果millis<0就抛出异常。

2.如果millis=0就判断当前调用join方法的线程是否存活。如果存活就不停地调用wait(0)方法。在一个实例Java对象上调用wait方法,那么当前线程就会从执行状态转变成等待状态,同时释放在实例对象上的锁,直到其它线程在刚才那个实例对象上调用notify方法并且释放实例对象上的锁,那么刚才那个当前线程才会再次获取实例对象锁并且继续执行。wait的作用是让“当前线程”等待,而这里的“当前线程”是指当前运行的线程。虽然是调用子线程的wait()方法,但是它是通过“主线程”去调用的;所以,休眠的是主线程,而不是“子线程”!

3.如果millis>0,先判断线程是否存活,然后调用wait方法。然后根据millis - now来调用wait方法,因为当调用t2.join时正在运行的是线程t1,所有主线程t1进入millis - now的阻塞状态,当时间超过millis - now毫秒是主线程将被唤醒。

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;
            }
        }
    }

这里解释一下wait()和isAlive()方法的来源。

isAlive方法被native关键字标注了,表明了要调用其他语言进行协助。所以isAlive方法体不用Java来写,wait方法也如此

public final native boolean isAlive();

wait方法也被native关键字标注了,表明了要调用其他语言进行协助。

public final native void wait(long timeoutMillis) throws InterruptedException;