Java中實現(xiàn)線程的創(chuàng)建和啟動的方法
1. 線程的生命周期
Java 線程有明確的生命周期狀態(tài):


- 新建狀態(tài)(New):創(chuàng)建 Thread 對象后(如 new MyThread()),線程尚未啟動
- 可運行狀態(tài)(Runnable):調(diào)用 start() 后,線程進(jìn)入就緒隊列
- 運行狀態(tài)(Running):線程獲得 CPU 時間片,執(zhí)行 run() 方法
- 終止?fàn)顟B(tài)(Terminated):run() 執(zhí)行完畢
關(guān)鍵點:只有調(diào)用 start() 才能使線程從 New 進(jìn)入 Runnable 狀態(tài)!
2. start() vs run() 的本質(zhì)區(qū)別
| 方法 | 工作方式 | 線程數(shù)量 | 執(zhí)行位置 |
|---|---|---|---|
start() | JVM 創(chuàng)建新線程并執(zhí)行 run() | 新線程 | 獨立調(diào)用棧 |
run() | 直接調(diào)用普通方法 | 當(dāng)前線程 | 當(dāng)前線程調(diào)用棧 |
錯誤示例:
Thread thread = new Thread(() -> System.out.println("執(zhí)行中"));
thread.run(); // 錯誤!在主線程同步執(zhí)行
正確示例:
thread.start(); // 正確!啟動新線程異步執(zhí)行
3. 為什么必須通過 start() 啟動線程?
(1) 資源分配
調(diào)用 start() 時,JVM 會:
- 為線程分配獨立調(diào)用棧(Stack)
- 注冊線程到系統(tǒng)調(diào)度器
- 觸發(fā)操作系統(tǒng)級別的線程創(chuàng)建
(2) 異步執(zhí)行
start()使任務(wù)在后臺異步執(zhí)行,不阻塞當(dāng)前線程- 直接調(diào)用
run()會同步執(zhí)行,阻塞當(dāng)前線程
(3) 線程調(diào)度控制
只有通過 start() 啟動的線程才能被:
- 線程調(diào)度器管理(優(yōu)先級、時間片分配)
- 正確響應(yīng)中斷(
interrupt()) - 加入線程池統(tǒng)一管理
(4) 狀態(tài)合規(guī)性
多次調(diào)用 start() 會拋出 IllegalThreadStateException,而 run() 可重復(fù)調(diào)用。這保證了線程狀態(tài)機(jī)的正確性。
4. 底層機(jī)制
當(dāng)調(diào)用 start() 時:
public synchronized void start() {
if (threadStatus != 0) // 檢查狀態(tài)是否為NEW
throw new IllegalThreadStateException();
group.add(this); // 加入線程組
boolean started = false;
try {
start0(); // 關(guān)鍵!調(diào)用本地方法
started = true;
} finally {
// ...錯誤處理
}
}
private native void start0(); // JVM實現(xiàn)的本地方法
start0()是native方法,由 JVM 通過操作系統(tǒng) API 創(chuàng)建真實線程- 新線程創(chuàng)建后自動執(zhí)行
run()方法
5. 實際應(yīng)用場景
假設(shè)需要同時下載3個文件:
// 錯誤方式(順序下載)
new DownloadTask("url1").run(); // 阻塞主線程
new DownloadTask("url2").run(); // 等待前一個完成
new DownloadTask("url3").run();
// 正確方式(并行下載)
new Thread(new DownloadTask("url1")).start(); // 異步
new Thread(new DownloadTask("url2")).start(); // 異步
new Thread(new DownloadTask("url3")).start(); // 異步
結(jié)論
| 操作 | 結(jié)果 | 是否啟動線程 |
|---|---|---|
new Thread() | 創(chuàng)建線程對象(NEW狀態(tài)) | ? |
start() | 啟動線程(進(jìn)入RUNNABLE狀態(tài)) | ? |
run() | 普通方法調(diào)用 | ? |
必須調(diào)用 start() 才能:
- 創(chuàng)建真正的操作系統(tǒng)線程
- 實現(xiàn)任務(wù)異步執(zhí)行
- 符合線程生命周期規(guī)范
- 利用多核CPU實現(xiàn)并行計算
直接調(diào)用 run() 只是普通方法調(diào)用,完全違背了多線程的設(shè)計目的!
到此這篇關(guān)于Java中實現(xiàn)線程的創(chuàng)建和啟動的方法的文章就介紹到這了,更多相關(guān)Java線程創(chuàng)建和啟動內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實現(xiàn)附件預(yù)覽(openoffice+swftools+flexpaper)實例
本篇文章主要介紹了java實現(xiàn)附件預(yù)覽(openoffice+swftools+flexpaper)實例,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2016-10-10
java獲取鍵盤輸入的數(shù)字,并進(jìn)行排序的方法
今天小編就為大家分享一篇java獲取鍵盤輸入的數(shù)字,并進(jìn)行排序的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-07-07
Java讀取Oracle大字段數(shù)據(jù)(CLOB)的2種方法
這篇文章主要介紹了Java讀取Oracle大字段數(shù)據(jù)(CLOB)的2種方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-04-04
SpringCloud Alibaba Seata (收藏版)
Seata是一款開源的分布式事務(wù)解決方案,致力于在微服務(wù)架構(gòu)在提供高性能和簡單一樣的分布式事務(wù)服務(wù)。這篇文章主要介紹了SpringCloud Alibaba Seata 的相關(guān)知識,需要的朋友可以參考下2020-10-10
Java基礎(chǔ)之隱式轉(zhuǎn)換vs強(qiáng)制轉(zhuǎn)換
這篇文章主要介紹了Java基礎(chǔ)之隱式轉(zhuǎn)換vs強(qiáng)制轉(zhuǎn)換的相關(guān)資料,需要的朋友可以參考下2015-12-12

