並發編程之執行緒屬性
- 2019 年 10 月 4 日
- 筆記
本文主要討論執行緒的各種屬性,包括:主執行緒、執行緒優先順序、守護執行緒、執行緒組、處理未捕獲異常的處理器。
public class Priority { @Test public void defaultPriority(){ int mainThreadPriority = Thread.currentThread().getPriority(); System.out.println("default priority is "+mainThreadPriority);//5 } @Test public void extendFather(){ Thread mainThread = Thread.currentThread(); mainThread.setPriority(4); int mainThreadPriority = mainThread.getPriority(); System.out.println("main thread's priority is "+mainThreadPriority);//4 Thread t1 = new Thread(() -> System.out.println(Thread.currentThread().getName()),"t1"); System.out.println("t1 thread's priority is "+t1.getPriority());//4 } }
相關API:
void setPriority(int priority) 設置執行緒優先順序 int getPriority() 獲取執行緒優先順序
守護執行緒
void setDaemon(boolean isdaemon) 設置執行緒為守護執行緒或者用戶執行緒。該方法必須在執行緒啟動之前調用。 boolean isDaemon() 判斷該執行緒是否為後台執行緒
執行緒組
public class ThreadGroupDemo { public static void print(){ Thread thread = Thread.currentThread(); System.out.println(thread.getThreadGroup().getName()+"-"+thread.getName()); } public static void main(String[] args) { ThreadGroup group = new ThreadGroup("Print Group"); new Thread(group, ThreadGroupDemo::print, "t1").start(); new Thread(group, ThreadGroupDemo::print, "t2").start(); group.list(); } }
未捕獲異常處理器
執行緒的run()不能拋出任何被檢測的異常【因為Runnable介面中定義的run()沒有拋出異常,所以重寫run()時,不允許拋出異常,可以使用try-catch捕獲異常】。但是,如果不被檢測的異常沒有使用try-catch處理,發生異常時會導致執行緒死亡。(比如下面這個例子,控制台並沒有輸出“Endind?”)
public class UncaughtExceptionDemo implements Runnable{ @Override public void run() { int i = 1/0; System.out.println("Endind?"); } public static void main(String[] args) { ExecutorService service = Executors.newFixedThreadPool(1); service.execute(new UncaughtExceptionDemo()); } } /* log: ... Exception in thread "pool-1-thread-1" java.lang.ArithmeticException: / by zero at concurrency.attributes.UncaughtExceptionDemo.run(UncaughtExceptionDemo.java:10) at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) at java.lang.Thread.run(Thread.java:748) */
而且,在run中手動拋出了一個運行時異常,在main中使用try-catch處理異常,並未生效。(如下例,控制台並未輸出”Current thread occurs exception.”)
public class UncaughtExceptionDemo implements Runnable{ @Override public void run() { int i = 1/0; } public static void main(String[] args) { try{ ExecutorService service = Executors.newFixedThreadPool(1); service.execute(new UncaughtExceptionDemo()); } catch (RuntimeException e){ System.out.println("Current thread occurs exception."); } } } /* ... Exception in thread "pool-1-thread-1" java.lang.ArithmeticException: / by zero at concurrency.attributes.UncaughtExceptionDemo.run(UncaughtExceptionDemo.java:10) at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) at java.lang.Thread.run(Thread.java:748) */
那麼,應該如何處理執行緒中拋出的異常呢?事實上,異常發生後,在執行緒死亡之前,異常會被傳遞到一個用於未捕獲異常的處理器。可以使用該處理器來處理執行緒中拋出的異常。
import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class ThreadExceptionResolve implements Runnable{ @Override public void run() { int i = 1/0; System.out.println("Endind?"); } public static void main(String[] args) { //2. 為所有的執行緒設置“異常處理器” Thread.setDefaultUncaughtExceptionHandler(new MyUncaughtExceptionHandler()); //3. 創建並執行執行緒 ExecutorService service = Executors.newFixedThreadPool(1); service.execute(new ThreadExceptionResolve()); } } //1. 定義符合執行緒異常處理器規範的“異常處理器” // 該處理器必須屬於一個實現Thread.UncaughtExceptionHandler介面的類 class MyUncaughtExceptionHandler implements Thread.UncaughtExceptionHandler { @Override public void uncaughtException(Thread t, Throwable e) { System.out.println("The acculation is error."); } } /* ... The acculation is error. */
安裝處理器的方式有兩種:可以使用Thread的靜態方法setDefaultUncaughtExceptionHandler(UncaughtExceptionHandler eh)為所有執行緒設置默認處理器,也可以使用Thread的實例方法setUncaughtExceptionHandler(UncaughtExceptionHandler eh)為摸一個執行緒設定處理器。如果執行緒沒有安裝處理器,此時的處理器就是該執行緒的ThreadGroup對象。
//為所有執行緒設置默認處理器 Thread.setDefaultUncaughtExceptionHandler(handler); //為指定執行緒設置處理器 Thread mainThread = Thread.currentThread(); mainThread.setUncaughtExceptionHandler(handler);