Java多執行緒並發01——執行緒的創建與終止,你會幾種方式

  • 2020 年 3 月 15 日
  • 筆記

本文開始將開始介紹 Java 多執行緒與並發相關的知識,多謝各位一直以來的關注與支援。關注我的公眾號「Java面典」了解更多 Java 相關知識點。

執行緒的創建方式

在 Java 中,用戶常用的主動創建執行緒的方式有三種,分別是 繼承 Thread 類實現 Runnable 介面通過Callable 和 Future

繼承 Thread 類

  • 定義 Thread 類的子類,並重寫該類的 run 方法;
  • 調用執行緒對象的 start() 方法來啟動該執行緒。

通過繼承 Thread 實現的執行緒類,多個執行緒間無法共享執行緒類的實例變數(需要創建不同 Thread 對象)。

/**   * 通過繼承Thread實現執行緒   */  public class MyThread extends Thread {      public void run() {          System.out.println("MyThread.run()");      }  }    MyThread myThread = new MyThread();  myThread.start();

實現 Runnable 介面

  • 如果自己的類已經 extends 另一個類,就無法直接 extends Thread,此時,可以實現一個 Runnable 介面;
  • 調用執行緒對象的start()方法來啟動該執行緒。
/**   * 通過實現Runnable介面實現的執行緒類   */  public class RunnableTest implements Runnable {      @Override      public void run() {          System.out.println("RunnableTest.run()");      }        public static void main(String[] args) {          RunnableTest runnableTest = new RunnableTest() ;          Thread thread = new Thread(runnableTest);          thread.start();      }  }

通過 Callable、Future

從 Thread 和 Runnable 兩種方式可以看出,兩種方式都不支援返回值,且不能聲明拋出異常

而 Callable 介面則實現了此兩點,Callable 介面如同 Runable 介面的升級版,其提供的 call() 方法將作為執行緒的執行體,同時允許有返回值。

但是 Callable 對象不能直接作為 Thread 對象的 target,我們可以使用 FutureTask 類來包裝 Callable 對象,該 FutureTask 對象封裝了該 Callable 對象的 call() 方法的返回值。

import java.util.concurrent.Callable;  import java.util.concurrent.FutureTask;    public class CallableTest {      public static void main(String[] args) {          CallableTest callableTest = new CallableTest() ;          //因為Callable介面是函數式介面,可以使用Lambda表達式          FutureTask<String> task = new FutureTask<Integer>((Callable<String>)()->{            System.out.println("FutureTask and Callable");            return "hello word";          });           try{             System.out.println("子執行緒返回值 : " + task.get());          } catch (Exception e){             e.printStackTrace();          }      }  }

執行緒的終止方式

執行緒除了正常結束外,還可以通過特定方式終止執行緒,終止執行緒常用的方式有以下三種:使用退出標誌退出執行緒、** Interrupt 方法結束執行緒stop 方法終止執行緒**。

使用退出標誌退出執行緒

最常使用的方式其實現方式是:定義一個 boolean 型的標誌位,在執行緒的 run() 方法中根據這個標誌位是 true 還是 false 來判斷是否退出,這種情況一般是將任務放在 run() 方法中的一個 while 循環中執行的。

public class ThreadSafe extends Thread {      public volatile boolean exit = false;      public void run() {          while (!exit){              //do work          }      }        public static void main(String[] args) throws Exception  {          ThreadFlag thread = new ThreadFlag();          thread.start();          sleep(5000); // 主執行緒延遲5秒          thread.exit = true;  // 終止執行緒thread          thread.join();          System.out.println("執行緒退出!");      }  }

Interrupt 方法結束執行緒

使用 interrupt() 方法來中斷執行緒有兩種情況:

  1. 執行緒處於阻塞狀態。如使用了 sleep,同步鎖的 wait,socket 中的 receiver,accept 等方法時,會使執行緒處於阻塞狀態。使用 interrupt 方法結束執行緒的時候,一定要先捕獲 InterruptedException 異常之後通過 break 來跳出循環,才能正常結束 run 方法。
public class ThreadInterrupt extends Thread {      public void run()  {          try {              sleep(50000);  // 延遲50秒          }          catch (InterruptedException e) {              System.out.println(e.getMessage());          }      }      public static void main(String[] args) throws Exception  {          Thread thread = new ThreadInterrupt();          thread.start();          System.out.println("在50秒之內按任意鍵中斷執行緒!");          System.in.read();          thread.interrupt();          thread.join();          System.out.println("執行緒已經退出!");      }  }
  1. 執行緒未處於阻塞狀態。使用 isInterrupted() 判斷執行緒的中斷標誌來退出循環。當使用 interrupt() 方法時,中斷標誌就會置 true,和使用自定義的標誌來控制循環是一樣的道理。
public class ThreadSafe extends Thread {      public void run() {          while (!isInterrupted()) { //非阻塞過程中通過判斷中斷標誌來退出              try {                  Thread.sleep(5*1000);//阻塞過程捕獲中斷異常來退出              } catch (InterruptedException e) {                  e.printStackTrace();                  break;//捕獲到異常之後,執行 break 跳出循環              }          }      }  }

stop 方法終止執行緒

使用 stop 方法可以強行終止正在運行或掛起的執行緒。我們可以使用如下的程式碼來終止執行緒:

thread.stop();

採用 stop 是不安全的,主要影響點如下:

  1. thread.stop() 調用之後,創建子執行緒的執行緒就會拋出 ThreadDeatherror 的錯誤;
  2. 調用 stop 會釋放子執行緒所持有的所有鎖。導致了該執行緒所持有的所有鎖的突然釋放(不可控制),那麼被保護數據就有可能呈現不一致性。

總結

  • 執行緒創建:推薦使用 Runnable 或者 Callable 方式創建執行緒,相比繼承,介面實現可以更加靈活,不會受限於Java的單繼承機制。
  • 執行緒終止:執行緒終止推薦使用 標誌位 或 Interrupt 方式終止,stop 方式對執行緒不安全,易導致數據不一致。