线程中处理异常是个头疼的问题,在异步的代码中,如何将异常捕获。

捕获异常后,将异常反馈给开发者,或最终用户。

一、将异常反馈给开发者,一般来说,在日志中打印异常日志即可;

二、将异常反馈给最终用户,一般来说,在页面上弹出提示框即可。

将异常打印到日志中(UncaughtExceptionHandler方式处理异常)

在线程当中遇到异常,需要捕捉并打印日志,我们可以考虑使用UncaughtExceptionHandler的回调。

以下是一个简单的测试例子。

有一个线程,方法体中会抛出异常

package com.nicchagil.study.thread.thinking.No15线程异常的捕捉;

public class ExceptionThread implements Runnable {

    @Override
public void run() {
throw new NullPointerException("故意抛出的异常。");
} }

自定义的异常处理器,这里所做的操作仅是在控制台打印异常,告诉开发者(并非用户哦)有这么个异常发生

package com.nicchagil.study.thread.thinking.No15线程异常的捕捉;

import java.lang.Thread.UncaughtExceptionHandler;

public class MyUncaughtExceptionHandler implements UncaughtExceptionHandler {

    @Override
public void uncaughtException(Thread t, Throwable throwable) {
System.out.println("Print exception @ MyUncaughtExceptionHandler -> " + throwable);
} }

为设置方便,通常可以有一个线程工厂(当然也可以直接设置,不通过工厂了),已帮忙批量以通用的参数设置产生线程。见调用类可见,有触发方法使用到此工厂,也有直接实例化线程的。

package com.nicchagil.study.thread.thinking.No15线程异常的捕捉;

import java.util.concurrent.ThreadFactory;

public class HandlerThreadFactory implements ThreadFactory {

    @Override
public Thread newThread(Runnable r) {
Thread t = new Thread(r);
t.setUncaughtExceptionHandler(new MyUncaughtExceptionHandler());
return t;
} }

触发类,这里介绍两种设置方式(一直接设置;一通过工厂设置),还有一种错误的设置方式,以供借镜。

package com.nicchagil.study.thread.thinking.No15线程异常的捕捉;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; public class Call { public static void main(String[] args) {
/* Work */
callByWay1(); /* Work */
callByWay2(); /* Un-work */
callByWay3();
} /**
* 原始的调用方式,在Thread对象设置异常处理器
*/
public static void callByWay1() {
try {
Thread t = new Thread(new ExceptionThread());
t.setUncaughtExceptionHandler(new MyUncaughtExceptionHandler());
t.start();
} catch (Throwable t) {
System.out.println("Print exception @ Call.java -> " + t);
}
} /**
* 使用工厂设置异常处理器的调用方式
*/
public static void callByWay2() {
try {
ExecutorService es = Executors.newCachedThreadPool(new HandlerThreadFactory());
es.execute(new ExceptionThread());
} catch (Throwable t) {
System.out.println("Print exception @ Call.java -> " + t);
}
} /**
* 实例化Thread后,传入ExecutorService调用。(此方法不行,原因见代码内注释)
*/
public static void callByWay3() {
try {
Thread t = new Thread(new ExceptionThread());
t.setUncaughtExceptionHandler(new MyUncaughtExceptionHandler()); ExecutorService es = Executors.newCachedThreadPool();
/* 此方法不行,异常处理器虽在前面设置,但由于es.execute(t)的入参为Runnable,故异常处理器遗失 */
es.execute(t);
} catch (Throwable t) {
System.out.println("Print exception @ Call.java -> " + t);
}
} }

日志如下,可见:

  1. 如果正确设置了异常处理器,将会触发异常处理器。我们的异常处理器中书写的逻辑是打印日志,所以,日志打印出来了。
  2. 无论有无设置异常处理器,异常都不会被直接的catch块捕捉和打印
Print exception @ MyUncaughtExceptionHandler -> java.lang.NullPointerException: 故意抛出的异常。
Print exception @ MyUncaughtExceptionHandler -> java.lang.NullPointerException: 故意抛出的异常。
Exception in thread "pool-1-thread-1" java.lang.NullPointerException: 故意抛出的异常。
at com.nicchagil.study.thread.thinking.No15线程异常的捕捉.ExceptionThread.run(ExceptionThread.java:7)
at java.lang.Thread.run(Thread.java:619)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
at java.lang.Thread.run(Thread.java:619)

将异常提示给最终用户(捕获异常)

告诉最终用户有异常发生,则需要捕获线程执行过程中产生的异常,捕获异常后传递给页面进行提示。

执行线程,并捕获异常:

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future; public class MyThread implements Callable<String> { @Override
public String call() throws Exception {
if (1 == 1) {
throw new RuntimeException("模拟的一个异常");
} return "hello world";
} public static void main(String[] args) {
ExecutorService es = Executors.newCachedThreadPool(); try {
Future<String> f = es.submit(new MyThread());
System.out.println(f.get());
} catch (Throwable t) {
System.out.println("捕获异常 : " + t.getMessage());
}
} }

日志:

看到“捕获异常 : java.lang.RuntimeException: 模拟的一个异常”就知道异常被捕捉了,剩下的就看你如何传到JSP了,哈哈哈。

捕获异常 : java.lang.RuntimeException: 模拟的一个异常

Runnable的线程也是可以的:

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future; public class MyThread implements Runnable { @Override
public void run() {
System.out.println(Thread.currentThread().getName() + " run.");
throw new RuntimeException("模拟的一个异常");
} public static void main(String[] args) {
ExecutorService es = Executors.newCachedThreadPool(); try {
Future<?> f = es.submit(new MyThread());
System.out.println(f.get());
} catch (Throwable t) {
System.out.println("捕获异常 : " + t.getMessage());
}
} }

【Thread】多线程的异常处理?的更多相关文章

  1. 多线程的异常处理、线程取消、临时变量、lock

    异步多线程的异常,抓不到,因为是在子线程执行. #region 多线程的异常处理.线程取消.临时变量.lock { try { List<Task> list = new List< ...

  2. java 多线程 线程组ThreadGroup;多线程的异常处理。interrupt批量停止组内线程;线程组异常处理

    1,线程组定义: 线程组存在的意义,首要原因是安全.java默认创建的线程都是属于系统线程组,而同一个线程组的线程是可以相互修改对方的数据的.但如果在不同的线程组中,那么就不能"跨线程组&q ...

  3. PowerBuilder编程新思维2:嵌入(Thread多线程)

    PowerBuilder编程新思维2:嵌入(Thread多线程) 在PB中使用多线程,在网上有大量的文章介绍.不过深入研究并试着给出更易用的模型的,目前还只有"路人甲cw"的一篇& ...

  4. Ptypes一个开源轻量级的c++库,包括对一些I/O操作、网络通信、多线程和异常处理的封装

    C++开源项目入门级:Ptypes    Ptypes一个开源轻量级的c++库,包括对一些I/O操作.网络通信.多线程和异常处理的封装.虽然代码有限,包括的内容不少,麻雀虽小,五脏俱全.    提高: ...

  5. C++ std::thread 多线程中的异常处理

    环境: VS2019 包含头文件: #include <iostream>#include<thread>#include<exception> 线程函数采用try ...

  6. .NET进阶篇06-async异步、thread多线程1

    知识需要不断积累.总结和沉淀,思考和写作是成长的催化剂 异步多线程挺大一块内容,既想拆开慢慢学,又想一股脑全倒出.纠结再三,还是拆开吃透,也不至于篇幅过长,劝退许多人 本篇先做一个概述,列明一些基本概 ...

  7. .NET进阶篇06-async异步、thread多线程2

    知识需要不断积累.总结和沉淀,思考和写作是成长的催化剂 内容目录 一.线程Thread1.生命周期2.后台线程3.静态方法1.线程本地存储2.内存栅栏4.返回值二.线程池ThreadPool1.工作队 ...

  8. C# 多线程总结 异常处理 线程取消 锁(lock)

    那么什么时候能用多线程? 任务能并发的时候 多线程能干嘛?提升速度/优化用户体验 网站首页:A数据库 B接口 C分布式服务 D搜索引擎,适合多线程并发,都完成后才能返回给用户,需要等待WaitAll列 ...

  9. JUC学习笔记--Thread多线程基础

    实现多线程的两种方法 java 实现多线程通过两种方式1.继承Thread类 ,2.实现Runnable接口 class Newthead extends Thread{ public void ru ...

随机推荐

  1. spark读取hdfs数据本地性异常

    在分布式计算中,为了提高计算速度,数据本地性是其中重要的一环. 不过有时候它同样也会带来一些问题. 一.问题描述 在分布式计算中,大多数情况下要做到移动计算而非移动数据,所以数据本地性尤其重要,因此我 ...

  2. paper 9:SVM番外篇:支持向量机系列六:Duality —— 关于 dual 问题推导的一些补充理论。

    在之前关于 support vector 的推导中,我们提到了 dual ,这里再来补充一点相关的知识.这套理论不仅适用于 SVM 的优化问题,而是对于所有带约束的优化问题都适用的,是优化理论中的一个 ...

  3. C# 把字符串类型日期转换为日期类型

    方法一:Convert.ToDateTime(string) string格式有要求,必须是yyyy-MM-dd hh:mm:ss ================================== ...

  4. js获取url的参数值

    var match = new RegExp('[?&]voucherSn=([^&]*)').exec("http://m.v3beta.tootoo.cn/index.p ...

  5. RobotFramework 安装配置(一)

    服务器接口的测试框架的选择,最后选中了 RobotFramework ,原因一:能有效的管理测试用例,,支持批量执行,能实现关键字驱动或者数据驱动.原因二:支持测试人员可以使用Python和java创 ...

  6. C#和JavaScript交互(asp.net前台和后台互调)总结 (转)

    http://www.cnblogs.com/poleices/archive/2011/02/24/1963727.html C#代码与javaScript函数的相互调用: 1.如何在JavaScr ...

  7. equals 与 == 的区别和用法(C# & Java)【转】

    转至http://www.cnblogs.com/beeone/archive/2011/04/25/2026674.html public class TestString { public sta ...

  8. Angularjs之表单实例(三)

    正确引用js css文件后可运行 <!DOCTYPE html> <html ng-app='myApp'> <head> <title>Bootstr ...

  9. gerrit-git

    解释为什么gerrit中的push是需要用refs/for/master http://stackoverflow.com/questions/10461214/why-is-git-push-ger ...

  10. 【python cookbook】【字符串与文本】15.给字符串中的变量名做插值处理

    问题:想创建一个字符串,其中嵌入的变量名称会以变量的字符串值形式替换掉 解决方法:str.format().str.format_map() >>> s = '{name} has ...