- 相關推薦
java基本教程之join方法的詳解
本章涉及到的內容包括:
1. join()介紹
2. join()源碼分析(基于JDK1.7.0_40)
3. join()示例
1. join()介紹
join() 定義在Thread.java中。
join() 的作用:讓“主線程”等待“子線程”結束之后才能繼續(xù)運行。這句話可能有點晦澀,我們還是通過例子去理解:
復制代碼 代碼如下:
// 主線程
public class Father extends Thread {
public void run() {
Son s = new Son();
s.start();
s.join();
...
}
}
// 子線程
public class Son extends Thread {
public void run() {
...
}
}
說明:
上面的有兩個類Father(主線程類)和Son(子線程類)。因為Son是在Father中創(chuàng)建并啟動的,所以,Father是主線程類,Son是子線程類。
在Father主線程中,通過new Son()新建“子線程s”。接著通過s.start()啟動“子線程s”,并且調用s.join()。在調用s.join()之后,Father主線程會一直等待,直到“子線程s”運行完畢;在“子線程s”運行完畢之后,Father主線程才能接著運行。 這也就是我們所說的“join()的作用,是讓主線程會等待子線程結束之后才能繼續(xù)運行”!
2. join()源碼分析(基于JDK1.7.0_40)
復制代碼 代碼如下:
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;
}
}
}
說明:
從代碼中,我們可以發(fā)現。當millis==0時,會進入while(isAlive())循環(huán);即只要子線程是活的,主線程就不停的等待。
我們根據上面解釋join()作用時的代碼來理解join()的用法!
問題:
雖然s.join()被調用的地方是發(fā)生在“Father主線程”中,但是s.join()是通過“子線程s”去調用的join()。那么,join()方法中的isAlive()應該是判斷“子線程s”是不是Alive狀態(tài);對應的wait(0)也應該是“讓子線程s”等待才對。但如果是這樣的話,s.join()的作用怎么可能是“讓主線程等待,直到子線程s完成為止”呢,應該是讓"子線程等待才對(因為調用子線程對象s的wait方法嘛)"?
答案:wait()的作用是讓“當前線程”等待,而這里的“當前線程”是指當前在CPU上運行的線程。所以,雖然是調用子線程的wait()方法,但是它是通過“主線程”去調用的;所以,休眠的是主線程,而不是“子線程”!
3. join()示例
在理解join()的作用之后,接下來通過示例查看join()的用法。
復制代碼 代碼如下:
// JoinTest.java的源碼
public class JoinTest{
public static void main(String[] args){
try {
ThreadA t1 = new ThreadA("t1"); // 新建“線程t1”
t1.start(); // 啟動“線程t1”
t1.join(); // 將“線程t1”加入到“主線程main”中,并且“主線程main()會等待它的完成”
System.out.printf("%s finishn", Thread.currentThread().getName());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
static class ThreadA extends Thread{
public ThreadA(String name){
super(name);
}
public void run(){
System.out.printf("%s startn", this.getName());
// 延時操作
for(int i=0; i<1000000; i++)
;
System.out.printf("%s finishn", this.getName());
}
}
}
運行結果:
復制代碼 代碼如下:
t1 start
t1 finish
main finish
結果說明:
運行流程如圖
(01) 在“主線程main”中通過 new ThreadA("t1") 新建“線程t1”。 接著,通過 t1.start() 啟動“線程t1”,并執(zhí)行t1.join()。
(02) 執(zhí)行t1.join()之后,“主線程main”會進入“阻塞狀態(tài)”等待t1運行結束!白泳程t1”結束之后,會喚醒“主線程main”,“主線程”重新獲取cpu執(zhí)行權,繼續(xù)運行。
【java基本教程之join方法的詳解】相關文章:
java教程之Java編程基礎09-12
Java數組的基本操作方法介紹08-14
java日期時間基本操作方法08-08
Java語言的基本特點07-30
Java類的基本構成08-28
java selenium基本教程09-27
Java語言基本概述10-15
Java語言基本特點06-01
JAVA多線程之線程間的通信方式解析07-14