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

Java 多線程啟動為什么調用 start() 方法而不是 run() 方法?

開發 后端
多線程在工作中多多少少會用到,啟動多線程調用的是 start() 方法,而不是 run() 方法,這是為什么呢?

 多線程在工作中多多少少會用到,啟動多線程調用的是 start() 方法,而不是 run() 方法,這是為什么呢?

[[325704]]

在探討這個問題之前,先來了解(復習)一些多線程的基礎知識~

線程的狀態

Java 中,定義了 6 種線程狀態,在 Thread 類可以找到:

 

  1. // 為了節約空間,我刪除了注釋 
  2. public enum State { 
  3.        NEW,//初始狀態 
  4.        RUNNABLE,//運行狀態 
  5.        BLOCKED,// 阻塞狀態 
  6.        WAITING,//等待狀態 
  7.        TIMED_WAITING,//超時等待狀態 
  8.        TERMINATED;//終止狀態 
  9.  } 

這 6 種狀態之間的關聯,可以看下面這張圖:

 

 

這張圖描述的還是非常詳細的,結合這張圖,來說說這幾種狀態分別代表著什么意思:

1、NEW 表示線程創建成功,但沒有運行,在 new Thread 之后,沒有 start 之前,線程都處于 NEW 狀態;

2、RUNNABLE 表示線程正在運行中,當我們運行 strat 方法,子線程被創建成功之后,子線程的狀態變成 RUNNABLE;

3、TERMINATED 表示線程已經運行結束,子線程運行完成、被打斷、被中止,狀態都會從 RUNNABLE 變成 TERMINATED;

4、BLOCKED 表示線程被阻塞,如果線程正好在等待獲得 monitor lock 鎖,比如在等待進入 synchronized 修飾的代碼塊或方法時,會從 RUNNABLE 變成 BLOCKED;

5、 WAITING 和 TIMED_WAITING 都表示等待,現在在遇到 Object#wait、Thread#join、 LockSupport#park 這些方法時,線程就會等待另一個線程執行完特定的動作之后,才能結 束等待,只不過 TIMED_WAITING 是帶有等待時間的;

優先級

優先級代表線程執行的機會的大小,優先級高的可能先執行,低的可能后執行。

在 Java 源碼中,優先級從低到高分別是 1 到 10,線程默認 new 出來的優先級都是 5,源碼如下:

 

  1. /** 
  2.   * The minimum priority that a thread can have. 
  3.   */ 
  4.  public final static int MIN_PRIORITY = 1; 
  5.  
  6. /** 
  7.   * The default priority that is assigned to a thread. 
  8.   */ 
  9.  public final static int NORM_PRIORITY = 5; 
  10.  
  11.  /** 
  12.   * The maximum priority that a thread can have. 
  13.   */ 
  14.  public final static int MAX_PRIORITY = 10; 

線程的創建方式

我們創建多線程有兩種方式,一種是繼承 Thread 類,另一種是實現 Runnable 接口。兩種方式的使用,如下所示:

1、繼承 Thread,成為 Thread 的子類

 

  1. public class MyThread extends Thread{ 
  2.     @Override 
  3.     public void run() { 
  4.         System.out.println("我是通過繼承 Thread 類實現的~"); 
  5.     } 
  6.  
  7.     public static void main(String[] args) { 
  8.         MyThread thread = new MyThread(); 
  9.         // 啟動線程 
  10.         thread.start(); 
  11.     } 

2、實現 Runnable 接口

 

  1. public class MyThread1 { 
  2.     public static void main(String[] args) { 
  3.         Thread thread = new Thread(new Runnable() { 
  4.             @Override 
  5.             public void run() { 
  6.                 System.out.println("我是通過 runnable 方式實現的~"); 
  7.             } 
  8.         }); 
  9.         // 啟動線程 
  10.         thread.start(); 
  11.     } 

不管使用哪一種方式,啟動線程都是thread.start()方法,如果你做過實驗的話,你會發現 thread.run()也可以執行,為什么就一定需要調用thread.start()方法呢?

先說說結論:首先通過對象.run()方法可以執行方法,但是不是使用的多線程的方式,就是一個普通的方法,要想實現多線程的方式,一定需要通過對象.start()方法。

想要弄明白一個問題,比較好的辦法就是從源碼入手,我們也從這兩個方法的源碼開始,先來看看 start 方法的源碼:

 

  1. public synchronized void start() { 
  2.     /** 
  3.      * This method is not invoked for the main method thread or "system" 
  4.      * group threads created/set up by the VM. Any new functionality added 
  5.      * to this method in the future may have to also be added to the VM. 
  6.      * 
  7.      * A zero status value corresponds to state "NEW"
  8.      */ 
  9.      // 沒有初始化,拋出異常 
  10.     if (threadStatus != 0) 
  11.         throw new IllegalThreadStateException(); 
  12.  
  13.     /* Notify the group that this thread is about to be started 
  14.      * so that it can be added to the group's list of threads 
  15.      * and the group's unstarted count can be decremented. */ 
  16.     group.add(this); 
  17.  // 是否啟動的標識符 
  18.     boolean started = false
  19.     try { 
  20.      // start0() 是啟動多線程的關鍵 
  21.      // 這里會創建一個新的線程,是一個 native 方法 
  22.      // 執行完成之后,新的線程已經在運行了 
  23.         start0(); 
  24.         // 主線程執行 
  25.         started = true
  26.     } finally { 
  27.         try { 
  28.             if (!started) { 
  29.                 group.threadStartFailed(this); 
  30.             } 
  31.         } catch (Throwable ignore) { 
  32.             /* do nothing. If start0 threw a Throwable then 
  33.               it will be passed up the call stack */ 
  34.         } 
  35.     } 

start 方法的源碼也沒幾行代碼,注釋也比較詳細,最主要的是 start0() 方法,這個后面在解釋。再來看看 run() 方法的源碼:

 

  1. @Override 
  2.    public void run() { 
  3.     // 簡單的運行,不會新起線程,target 是 Runnable 
  4.        if (target != null) { 
  5.            target.run(); 
  6.        } 
  7.    } 

run() 方法的源碼就比較簡單的,就是一個普通方法的調用,這也印證了我們上面的結論。

接下來我們就來說一說這個 start0() 這個方法,這個是真正實現多線程的關鍵,start0() 代碼如下:

 

  1. private native void start0(); 

start0 被標記成 native ,也就是本地方法,并不需要我們去實現或者了解,**為什么 start0() 會標記成 native **?

這個要從 Java 跨平臺說起,看下面這張圖:

 

 

start() 方法調用 start0() 方法后,該線程并不一定會立馬執行,只是將線程變成了可運行狀態(NEW ---> RUNNABLE)。具體什么時候執行,取決于 CPU ,由 CPU 統一調度。

我們又知道 Java 是跨平臺的,可以在不同系統上運行,每個系統的 CPU 調度算法不一樣,所以就需要做不同的處理,這件事情就只能交給 JVM 來實現了,start0() 方法自然就表標記成了 native。

最后,總結一下,Java 中實現真正的多線程是 start 中的 start0() 方法,run() 方法只是一個普通的方法。

責任編輯:華軒 來源: 互聯網平頭哥
相關推薦

2022-02-15 07:03:04

start 源碼run線程

2022-09-05 15:36:47

線程方法Java

2020-12-15 07:36:12

線程Start Run

2013-03-25 10:14:18

NginxApache

2019-04-19 11:56:48

框架AI開發

2012-10-10 16:52:21

CentOSDebianUbuntu

2021-10-30 19:57:00

HTTP2 HTTP

2021-08-14 09:04:58

TypeScriptJavaScript開發

2009-08-26 16:58:12

調用C# Thread

2010-03-16 09:19:22

Java多線程

2020-09-15 09:23:19

C++WindowsC#

2017-09-11 19:58:06

PostgreSQLMySQL數據庫

2020-06-02 14:17:55

QWER排列鍵盤打印機

2009-06-29 18:08:51

Java多線程join方法

2021-02-26 05:30:25

元素For-Each代碼

2023-11-02 08:20:54

SocketZygoteAndroid

2021-03-26 11:50:28

Linuxexals

2023-03-01 10:42:58

gRPC服務端設置

2021-06-30 12:47:12

標簽HTML分辨率

2015-09-10 09:30:54

Java多線程同步
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 羞羞的视频免费观看 | 天天操天天干天天爽 | 综合久久综合久久 | 一区二区三区国产 | 91亚洲国产亚洲国产 | 91porn在线观看| 国产精品中文字幕在线 | 中文字幕在线剧情 | 99精品欧美一区二区三区综合在线 | 奇米久久| 亚洲国产成人精品女人久久久 | 在线视频91 | 午夜不卡一区二区 | 国产视频久久久 | 欧美伊人影院 | 视频一二区 | 久久国产精品一区二区三区 | 国产良家自拍 | 91天堂| 色就干 | 久草欧美| 九九色综合 | 日韩欧美三区 | 美国十次成人欧美色导视频 | 天堂成人国产精品一区 | 龙珠z国语版在线观看 | 亚洲三区在线播放 | 久久久久1 | 国产成人一区二区三区 | 欧美精品成人一区二区三区四区 | 在线婷婷 | 五月激情综合网 | www狠狠爱com| 亚洲精品1 | 欧美成人精品在线观看 | 亚洲欧洲一区二区 | 一区二区久久 | 欧美日韩综合精品 | 日本成人一区二区 | 日本欧美国产在线 | 久久精品视频在线观看 |