Java如何主動(dòng)從當(dāng)前線程獲取異常信息
Java主動(dòng)從當(dāng)前線程獲取異常信息
使用場(chǎng)景
在單個(gè)方法內(nèi)主動(dòng)捕獲異常,并將異常的錯(cuò)誤棧信息以日志的方式打出來
寫法
當(dāng)前方法throw 了 異常,即改方法存在異常的情況,則可以使用如下方式獲取當(dāng)前線程中的異常:
// 主動(dòng)獲取當(dāng)前線程異常棧信息 StackTraceElement[] stackTraceElements = Thread.currentThread().getStackTrace(); // 以日志的方式打出 ?\n ?每一行錯(cuò)誤棧信息結(jié)束后換行 log.error(StringUtils.join(stackTraceElements, '\n'));
Java捕獲并處理線程異常:Thread及ThreadPoolExecutor線程池異常捕獲
通過Thread.UncaughtExceptionHandler捕獲線程異常
Thread.UncaughtExceptionHandler類的作用:捕獲并處理線程run方法拋出的異常。
使用示例
為單個(gè)線程設(shè)置異常捕獲
Thread.UncaughtExceptionHandler exceptionHandler = (t, e) -> {
? ? System.out.println("報(bào)錯(cuò)線程:" + t.getName());
? ? System.out.println("線程拋出的異常:" + e);
};
Thread thread = new Thread(() -> {
? ? throw new RuntimeException("throw a new Exception ! ");
});
thread.setUncaughtExceptionHandler(exceptionHandler); // 設(shè)置異常處理器
thread.start();如果項(xiàng)目中,全局的Thread線程處理異常的方式都相同,那么可以設(shè)置一個(gè)全局的異常捕獲類。
Thread.UncaughtExceptionHandler exceptionHandler = (t, e) -> {
? ? System.out.println("報(bào)錯(cuò)線程:" + t.getName());
? ? System.out.println("線程拋出的異常:" + e);
};
Thread.setDefaultUncaughtExceptionHandler(exceptionHandler);通過這種方式,后續(xù)的Thread都可以公用這個(gè)異常處理類。如果需要用其它方式處理異常時(shí),只需要實(shí)現(xiàn)1中的內(nèi)容即可。
部分源碼解析
UncaughtExceptionHandler源碼
// Thread.UncaughtExceptionHandler?
@FunctionalInterface
public interface UncaughtExceptionHandler {
? ? void uncaughtException(Thread t, Throwable e);
}Thread中的UncaughtExceptionHandler屬性
// 作用于當(dāng)個(gè)Thread線程 private volatile UncaughtExceptionHandler uncaughtExceptionHandler; // static修飾,其可以作用于所有Thread線程 private static volatile UncaughtExceptionHandler defaultUncaughtExceptionHandler;
在Thread中,會(huì)首先提供Thread私有的異常處理類,然后才是全局
public UncaughtExceptionHandler getUncaughtExceptionHandler() {
? ? return uncaughtExceptionHandler != null ?
? ? ? ? uncaughtExceptionHandler : group;
}實(shí)現(xiàn)原理
當(dāng)線程由于未捕獲的異常而即將終止時(shí),Java虛擬機(jī)將使用調(diào)用線程的getUncaughtExceptionHandler方法,以此來獲取UncaughtExceptionHandler類,并執(zhí)行其對(duì)異常的處理方法。
如果一個(gè)線程沒有顯式實(shí)現(xiàn)UncaughtExceptionHandler ,那么它的ThreadGroup對(duì)象將充當(dāng)它的UncaughtExceptionHandler類。
// Thread#getUncaughtExceptionHandler
public UncaughtExceptionHandler getUncaughtExceptionHandler() {
? ? return uncaughtExceptionHandler != null ?
? ? ? ? uncaughtExceptionHandler : group;
}ThreadGroup實(shí)現(xiàn)了UncaughtExceptionHandler類。
public class ThreadGroup implements Thread.UncaughtExceptionHandler {
?? ?// ……
?? ?public void uncaughtException(Thread t, Throwable e) {
?? ??? ?// 父級(jí)ThreadGroup的處理方法
?? ? ? ?if (parent != null) {
?? ? ? ? ? ?parent.uncaughtException(t, e);
?? ? ? ?} else {
?? ? ? ??? ?// Thread 的全局默認(rèn)處理方法
?? ? ? ? ? ?Thread.UncaughtExceptionHandler ueh =
?? ? ? ? ? ? ? ?Thread.getDefaultUncaughtExceptionHandler();
?? ? ? ? ? ?if (ueh != null) {
?? ? ? ? ? ? ? ?ueh.uncaughtException(t, e);
?? ? ? ? ? ?} else if (!(e instanceof ThreadDeath)) {
?? ? ? ? ? ? ? ?System.err.print("Exception in thread \""
?? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? + t.getName() + "\" ");
?? ? ? ? ? ? ? ?e.printStackTrace(System.err);
?? ? ? ? ? ?}
?? ? ? ?}
?? ?}
?? ?// ……
}ThreadPoolExecutor線程池異常捕獲
使用示例
要捕獲ThreadPoolExecutor線程池中的線程執(zhí)行異常,需要實(shí)現(xiàn)被protected修飾的方法afterExecute,在該方法里面處理異常即可。
// ThreadPoolExecutor#afterExecute
class ExtendedExecutor extends ThreadPoolExecutor {
? protected void afterExecute(Runnable r, Throwable t) {
? ? super.afterExecute(r, t);
? ? // 如果Runnable是Future類型,那么異常將會(huì)直接通過Future返回
? ? if (t == null && r instanceof Future<?>) {
? ? ? try {
? ? ? ? Object result = ((Future<?>) r).get();
? ? ? } catch (CancellationException ce) {
? ? ? ? ? t = ce;
? ? ? } catch (ExecutionException ee) {
? ? ? ? ? t = ee.getCause();
? ? ? } catch (InterruptedException ie) {
? ? ? ? ? Thread.currentThread().interrupt(); // ignore/reset
? ? ? }
? ? }
? ? // 非Future類型
? ? if (t != null)
? ? ? System.out.println(t);
? }
}注意:實(shí)現(xiàn)afterExecute方法時(shí),要正確嵌套多個(gè)覆蓋,子類通常應(yīng)在此方法的開頭調(diào)用super.afterExecute,以確保不會(huì)破壞其他父類方法的實(shí)現(xiàn)。
源碼解析
在ThreadPoolExecutor中,線程任務(wù)的實(shí)際執(zhí)行方法是runWorker,如下所示:
// ThreadPoolExecutor#runWorker
final void runWorker(Worker w) {
? ? Thread wt = Thread.currentThread();
? ? Runnable task = w.firstTask; // 實(shí)際提交的任務(wù)
? ? ? ? ? ? // …………
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? beforeExecute(wt, task); // task執(zhí)行前的操作
? ? ? ? ? ? ? ? Throwable thrown = null; // 異常信息保存
? ? ? ? ? ? ? ? try {
? ? ? ? ? ? ? ? ? ? task.run(); // 執(zhí)行任務(wù)
? ? ? ? ? ? ? ? } catch (RuntimeException x) {
? ? ? ? ? ? ? ? ? ? thrown = x; throw x;
? ? ? ? ? ? ? ? } catch (Error x) {
? ? ? ? ? ? ? ? ? ? thrown = x; throw x;
? ? ? ? ? ? ? ? } catch (Throwable x) {
? ? ? ? ? ? ? ? ? ? thrown = x; throw new Error(x);
? ? ? ? ? ? ? ? } finally {
? ? ? ? ? ? ? ? ? ? afterExecute(task, thrown); // task執(zhí)行后的操作,也就包括了異常的捕獲工作
? ? ? ? ? ? ? ? }
? ? ? ? ? ? } finally {
? ? ? ? ? ? ? ? task = null;
? ? ? ? ? ? ? ? w.completedTasks++;
? ? ? ? ? ? ? ? w.unlock();
? ? ? ? ? ? }
? ? ? ? // ……
}afterExecute方法在ThreadPoolExecutor,并沒有進(jìn)行任何操作,就是對(duì)異常的線程靜默處理
// ThreadPoolExecutor#afterExecute
protected void afterExecute(Runnable r, Throwable t) { }Callable類型的任務(wù),在執(zhí)行時(shí)會(huì)自己捕獲并維護(hù)執(zhí)行中的異常。
// AbstractExecutorService#submit
public <T> Future<T> submit(Callable<T> task) {
? ? if (task == null) throw new NullPointerException();
? ? RunnableFuture<T> ftask = newTaskFor(task);
? ? execute(ftask);
? ? return ftask;
}
// Callable任務(wù)會(huì)被封裝成FutureTask
protected <T> RunnableFuture<T> newTaskFor(Callable<T> callable) {
? ? return new FutureTask<T>(callable);
}在FutureTask的run方法中,他們內(nèi)部會(huì)將整個(gè)任務(wù)用try-catch給包起來。因此,也不會(huì)拋出Callable#run執(zhí)行的內(nèi)部異常。
// FutureTask#run
public void run() {
?? ?// ……
? ? try {
? ? ? ? Callable<V> c = callable;
? ? ? ? // ……
? ? ? ? ? ? // 這里將Callable的執(zhí)行過程產(chǎn)生的異常都捕獲了
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? result = c.call();
? ? ? ? ? ? ? ? ran = true;
? ? ? ? ? ? } catch (Throwable ex) {
? ? ? ? ? ? ? ? result = null;
? ? ? ? ? ? ? ? ran = false;
? ? ? ? ? ? ? ? setException(ex);
? ? ? ? ? ? }
? ? ? ? ? ? if (ran)
? ? ? ? ? ? ? ? set(result);
? ? ? ? }
? ? } finally {
// ……以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Maven 打包項(xiàng)目到私服 (deploy)的配置方法
這篇文章主要介紹了Maven 打包項(xiàng)目到私服 (deploy)的配置方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07
Mybatis有查詢結(jié)果但存不進(jìn)實(shí)體類的解決方案
這篇文章主要介紹了Mybatis有查詢結(jié)果但存不進(jìn)實(shí)體類的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-11-11
java interface的兩個(gè)經(jīng)典用法
這篇文章主要為大家詳細(xì)介紹了java interface的兩個(gè)經(jīng)典用法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-09-09
Java?使用geotools讀取tiff數(shù)據(jù)的示例代碼
這篇文章主要介紹了Java?通過geotools讀取tiff,一般對(duì)于tiff數(shù)據(jù)的讀取,都會(huì)借助于gdal,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-04-04
Java實(shí)現(xiàn)JSON與XML相互轉(zhuǎn)換的簡(jiǎn)明教程
Java實(shí)現(xiàn)復(fù)雜數(shù)據(jù)結(jié)構(gòu)(如嵌套對(duì)象、數(shù)組)在 JSON 與 XML 之間的相互轉(zhuǎn)換,可以使用 Jackson 和 Jackson XML 擴(kuò)展庫(kù)來完成,Jackson 是一個(gè)流行的 JSON 處理庫(kù),通過 Jackson 的 XML 擴(kuò)展庫(kù),可以實(shí)現(xiàn) JSON 和 XML 之間的轉(zhuǎn)換,需要的朋友可以參考下2024-08-08
SpringBoot Actuator監(jiān)控的項(xiàng)目實(shí)踐
本文主要結(jié)合 Spring Boot Actuator,跟大家一起分享微服務(wù)Spring Boot Actuator 的常見用法,方便我們?cè)谌粘V袑?duì)我們的微服務(wù)進(jìn)行監(jiān)控治理,感興趣的可以了解一下2024-01-01
8個(gè)Spring事務(wù)失效場(chǎng)景詳解
相信大家對(duì)Spring種事務(wù)的使用并不陌生,但是你可能只是停留在基礎(chǔ)的使用層面上。今天,我們就簡(jiǎn)單來說下Spring事務(wù)的原理,然后總結(jié)一下spring事務(wù)失敗的場(chǎng)景,并提出對(duì)應(yīng)的解決方案,需要的可以參考一下2022-12-12
Mybatis中強(qiáng)大的resultMap功能介紹
這篇文章主要給大家介紹了關(guān)于Mybatis中強(qiáng)大的resultMap功能的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Mybatis具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06

