成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

這么理解線程生命周期,是不是很簡單?

開發 后端
了解了它們的生命周期后,使用 Spring Bean 好比看到它們的行動軌跡,現在使用就一點都不慌了。

為什么要了解線程的生命周期?

之前寫過 Spring Bean 生命周期三部曲:

  1.  Spring Bean生命周期之緣起
  2.  Spring Bean生命周期之緣盡
  3.  Spring Aware 到底是什么?

有朋友留言說:“了解了它們的生命周期后,使用 Spring Bean 好比看到它們的行動軌跡,現在使用就一點都不慌了”。我和他一樣,了解事物的生命周期目的很簡單,唯【不慌】也

[[320328]]

Java 并發系列 已經寫了很多,從來還沒提起過那個它【Java線程生命周期】。有了前序理論圖文的鋪墊,在走進源碼世界之前,談論它的時機恰好到了。因為,編寫并發程序的核心之一就是正確的擺弄線程狀態

線程生命周期的幾種狀態

剛接觸線程生命周期時,我總是記不住,也理解不了他們的狀態,可以說是比較混亂,更別說它們之間是如何進行狀態轉換的了。原因是我把操作系統通用線程狀態和編程語言封裝后的線程狀態 概念混淆在一起了

操作系統通用線程狀態

個人覺得通用線程狀態更符合我們的思考習慣。其狀態總共有 5 種 (如下圖)。對于經常寫并發程序的同學來說,其嘴里經常念的都是操作系統中的這些通用線程狀態,且看

除去生【初始狀態】死【終止狀態】,其實只是三種狀態的各種轉換,聽到這句話是不是心情放松了很多呢?

為了更好的說明通用線程狀態和 Java 語言中的線程狀態,這里還是先對前者進行簡短的說明

初始狀態

線程已被創建,但是還不被允許分配CPU執行。注意,這個被創建其實是屬于編程語言層面的,實際在操作系統里,真正的線程還沒被創建, 比如 Java 語言中的 new Thread()。

可運行狀態

線程可以分配CPU執行,這時,操作系統中線程已經被創建成功了

運行狀態

操作系統會為處在可運行狀態的線程分配CPU時間片,被 CPU 臨幸后,處在可運行狀態的線程就會變為運行狀態

休眠狀態

如果處在運行狀態的線程調用某個阻塞的API或等待某個事件條件可用,那么線程就會轉換到休眠狀態,注意:此時線程會釋放CPU使用權,休眠的線程永遠沒有機會獲得CPU使用權,只有當等待事件出現后,線程會從休眠狀態轉換到可運行狀態

終止狀態

線程執行完或者出現異常 (被interrupt那種不算的哈,后續會說)就會進入終止狀態,正式走到生命的盡頭,沒有起死回生的機會

接下來就來看看你熟悉又陌生,面試又經常被問到的Java 線程生命周期吧

Java語言線程狀態

在 Thread 的源碼中,定義了一個枚舉類 State,里面清晰明了的寫了Java語言中線程的6種狀態:

  1.  NEW
  2.  RUNNABLE
  3.  BLOCKED
  4.  WAITING
  5.  TIMED_WAITING
  6.  TERMINATED

這里要做一個小調查了,你有查看過這個類和讀過其注釋說明嗎?(歡迎留言腳印哦)

[[320330]]

耳邊響起五環之歌,Java中線程狀態竟然比通用線程狀態的 5 種多1種,變成了 6 種。這個看似復雜,其實并不是你想的那樣,Java在通用線程狀態的基礎上,有裁剪,也有豐富,整體來說是少一種。再來看個圖,注意顏色區分哦

Java 語言中

  •  將通用線程狀態的可運行狀態和運行狀態合并為 Runnable,
  •  將休眠狀態細分為三種 (BLOCKED/WAITING/TIMED_WAITING); 反過來理解這句話,就是這三種狀態在操作系統的眼中都是休眠狀態,同樣不會獲得CPU使用權

看上圖右側【Java語言中的線程狀態】,進一步簡潔的說,除去線程生死,我們只要玩轉 RUNNABLE 和休眠狀態的轉換就可以了,編寫并發程序也多數是這兩種狀態的轉換。所以我們需要了解,有哪些時機,會觸發這些狀態轉換

遠看看輪廓, 近看看細節。我們將上面Java語言中的圖進行細化,將觸發的節點放到圖中 (這看似復雜的圖,其實三句話就能分解的,所以別慌),且看:

RUNNABLE與BLOCKED狀態轉換

當且僅有(just only)一種情況會從 RUNNABLE 狀態進入到 BLOCKED 狀態,就是線程在等待 synchronized 內置隱式鎖;如果等待的線程獲取到了 synchronized 內置隱式鎖,也就會從 BLOCKED 狀態變為 RUNNABLE 狀態了

注意:

上面提到,以操作系統通用狀態來看,線程調用阻塞式 API,會變為休眠狀態(釋放CPU使用權),但在JVM層面,Java線程狀態不會發生變化,也就是說Java線程的狀態依舊會保持在 RUNNABLE 狀態。JVM并不關心操作系統調度的狀態。在JVM看來,等待CPU使用權(操作系統里是處在可執行狀態)與等待I/O(操作系統是處在休眠狀態),都是等待某個資源,所以都歸入了RUNNABLE 狀態

    ​ —— 摘自《Java并發編程實戰》

RUNNABLE與WAITING狀態轉換

調用不帶時間參數的等待API,就會從RUNNABLE狀態進入到WAITING狀態;當被喚醒就會從WAITING進入RUNNABLE狀態

RUNNABLE與 TIMED-WAITING 狀態轉換

調用帶時間參數的等待API,自然就從 RUNNABLE 狀態進入 TIMED-WAITING 狀態;當被喚醒或超時時間到就會從TIMED_WAITING進入RUNNABLE狀態

看圖中的轉換 API 挺多的,其實不用擔心,后續分析源碼章節,自然就會記住的,現在有個印象以及知道狀態轉換的節點就好了

相信到這里,你看Java線程生命周期的眼神就沒那么迷惑了,重點就是RUNNABLE與休眠狀態的切換,接下來我們看一看,如何查看線程中的狀態,以及具體的代碼觸發點

如何查看線程處在什么狀態

程序中調用 getState() 方法

Thread 類中同樣存在 getState() 方法用于查看當前線程狀態,該方法就是返回上面提到的枚舉類 State

NEW

就是上面提到, 編程語言中特有的,通過繼承 Thread 或實現 Runnable 接口定義線程后,這時的狀態都是 NEW 

  1. Thread thread = new Thread(() -> {});  
  2. System.out.println(thread.getState()); 

RUNNABLE

調用了 start() 方法之后,線程就處在 RUNNABLE 狀態了 

  1. Thread thread = new Thread(() -> {});  
  2. thread.start();  
  3. //Thread.sleep(1000);  
  4. System.out.println(thread.getState()); 

BLOCKED

等待 synchronized 內置鎖,就會處在 BLOCKED 狀態 

  1. public class ThreadStateTest { 
  2.     public static void main(String[] args) throws InterruptedException {  
  3.         Thread t1 = new Thread(new DemoThreadB());  
  4.         Thread t2 = new Thread(new DemoThreadB());  
  5.         t1.start();  
  6.         t2.start(); 
  7.         Thread.sleep(1000);  
  8.         System.out.println((t2.getState()));  
  9.         System.exit(0); 
  10.     }  
  11.  
  12. class DemoThreadB implements Runnable {  
  13.     @Override  
  14.     public void run() {  
  15.         commonResource();  
  16.     }  
  17.     public static synchronized void commonResource() {  
  18.         while(true) {     
  19.          }  
  20.     }  

WAITING

調用線程的 join() 等方法,從 RUNNABLE 變為 WAITING 狀態 

  1. public static void main(String[] args) throws InterruptedException {  
  2.         Thread main = Thread.currentThread();  
  3.         Thread thread2 = new Thread(() -> {  
  4.             try {  
  5.                 Thread.sleep(1000);  
  6.             } catch (InterruptedException e) {  
  7.         Thread.currentThread().interrupt();  
  8.                 e.printStackTrace();  
  9.             }  
  10.             System.out.println(main.getState());  
  11.         });  
  12.         thread2.start();  
  13.         thread2.join();  
  14.     } 

TIMED-WAITING

調用了 sleep(long) 等方法,線程從 RUNNABLE 變為 TIMED-WAITING 狀態 

  1. public static void main(String[] args) throws InterruptedException {  
  2.         Thread thread3 = new Thread(() -> {  
  3.             try {  
  4.                 Thread.sleep(3000);  
  5.             } catch (InterruptedException e) {  
  6.         // 為什么要調用interrupt方法?  
  7.                 Thread.currentThread().interrupt();  
  8.                 e.printStackTrace();  
  9.             }  
  10.         });  
  11.         thread3.start();  
  12.         Thread.sleep(1000);  
  13.         System.out.println(thread3.getState());  
  14.     } 

TERMINATED

線程執行完自然就到了 TERMINATED 狀態了 

  1. Thread thread = new Thread(() -> {});  
  2. thread.start();  
  3. Thread.sleep(1000);  
  4. System.out.println(thread.getState()); 

以上是程序中查看線程,自己寫寫測試看看狀態還好,現實中的程序怎么可能允許你加這么多無用代碼,所以,翠花,上酸菜(jstack)

[[320331]]

jstack 命令查看

相信你聽說過這玩意,jstack 命令就比較強大了,不僅能查看線程當前狀態,還能看調用棧,鎖等線程棧信息

大家可以隨意寫一些程序,這里我用了上面 WAITING 狀態的代碼, 修改睡眠時間 Thread.sleep(100000),然后在終端按照下圖標示依次執行下圖命令 

更多功能還請大家自行查看,后續會單獨寫文章來教大家如何使用jstack查看線程棧信息

Arthas

這個利器,無須多言吧,線上找茬監控沒毛病,希望你可以靈活使用這個工具,攻克疑難雜癥

查看線程棧詳細信息,非常方便:https://alibaba.github.io/art...

相信你已經和Arthas確認了眼神

[[320332]]

關于線程生命周期狀態整體就算說完了,編寫并發程序時多問一問自己:

調用某個API會將你的線程置為甚么狀態?

多問自己幾次,自然就記住上面的圖了

靈魂追問

  1.  為什么調用 Thread.sleep, catch異常后,調用了Thread.currentThread().interrupt();
  2.  進入 BLOCKED只有一種情況,就是等待 synchronized 監視器鎖,那調用 JUC 中的 Lock.lock() 方法,如果某個線程等待這個鎖,這個線程狀態是什么呢?為什么?   
  1. public class ThreadStateTest {  
  2.        public static void main(String[] args) throws InterruptedException {  
  3.            TestLock testLock = new TestLock();  
  4.            Thread thread2 = new Thread(() -> {  
  5.                testLock.myTestLock();  
  6.            }, "thread2");  
  7.            Thread thread1 = new Thread(() -> {  
  8.                    testLock.myTestLock();  
  9.                }, "thread1");  
  10.            thread1.start();  
  11.            Thread.sleep(1000);  
  12.            thread2.start();  
  13.            Thread.sleep(1000);  
  14.            System.out.println("****" + (thread2.getState()));  
  15.            Thread.sleep(20000); 
  16.        }  
  17.    }  
  18.    @Slf4j  
  19.    class TestLock{  
  20.        private final Lock lock = new ReentrantLock();  
  21.        public void myTestLock(){  
  22.            lock.lock();  
  23.            try{  
  24.                Thread.sleep(10000);  
  25.                log.info("testLock status");  
  26.            } catch (InterruptedException e) {  
  27.                log.error(e.getMessage());  
  28.            } finally {  
  29.                lock.unlock();  
  30.            }  
  31.        }  
  32.    } 

      3.   synchronized 和 Lock 有什么區別?

參考

感謝前輩們總結的精華,自己所寫的并發系列好多都參考了以下資料

  •  Java 并發編程實戰
  •  Java 并發編程之美
  •  碼出高效
  •  Java 并發編程的藝術
  •  ......

我這面也在逐步總結常見的并發面試問題(總結ing......)答案整理好后會通知大家,請持續關注

 

 

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2009-06-24 10:47:55

JSF生命周期

2022-04-19 07:20:24

軟件開發安全生命周期SSDLC應用安全

2010-07-14 10:48:37

Perl線程

2009-06-18 13:32:39

Java線程生命周期

2012-01-16 09:00:56

線程

2009-06-29 18:03:15

Java多線程線程的生命周期

2010-07-14 10:59:15

Perl線程

2023-10-26 08:25:35

Java線程周期

2015-07-08 16:28:23

weak生命周期

2021-08-24 07:53:28

AndroidActivity生命周期

2012-06-20 10:29:16

敏捷開發

2013-08-19 17:03:00

.Net生命周期對象

2021-07-19 05:52:29

網絡生命周期網絡框架

2009-06-11 11:28:35

JSF生命周期

2012-04-28 13:23:12

Java生命周期

2014-07-16 13:39:30

Windows Pho

2013-08-01 09:40:51

Windows 8.1

2023-09-12 10:52:16

OpenSSL開源套件

2011-06-16 09:31:21

ActivityAndroid

2023-10-07 00:05:07

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 四虎最新| 久久久男人的天堂 | 日韩在线欧美 | 精品一区二区av | 亚洲久在线 | 91精品国产综合久久香蕉麻豆 | 91精品久久久久久久久 | 精品乱码一区二区 | 91久久精品日日躁夜夜躁国产 | 欧美日韩国产综合在线 | 国产精品毛片无码 | 国产真实乱对白精彩久久小说 | 亚洲国产精品精华素 | 99热在线观看精品 | 欧美日韩在线精品 | 欧美日韩在线免费 | 国产精品欧美一区二区三区不卡 | 蜜桃精品噜噜噜成人av | 99久久精品免费看国产四区 | 亚洲美女在线一区 | 久久久久www | 欧美精品一区二区在线观看 | 天天干天天操天天看 | 婷婷精品| 91在线看片 | 2021狠狠天天天 | 日韩精品在线播放 | 成人在线免费 | 日本视频在线 | 狠狠操狠狠干 | 国产成人免费 | 亚洲精品一区二区二区 | 91在线视频免费观看 | 日韩精品一区二区三区老鸭窝 | 欧美日韩三级 | 超级黄色一级片 | 午夜在线视频 | 国产91av视频 | 久久精品国产一区 | 可以在线看的黄色网址 | 亚洲美女视频 |