Java并发-异常处理

由于异常不能跨线程传递,这就导致不能在主线程中处理子线程的异常,就像下面这样

class ExceptionTask implements Runnable {
    @Override
    public void run() {
        System.out.println("Exception Task ");
        throw new RuntimeException("Thread Exception");
    }
}
public class ExceptionHandleTest {
    public static void main(String[] args) {
        try {
            Thread thread = new Thread(new ExceptionTask());
            thread.start();
        }catch (Exception ex) {
            ex.printStackTrace();
        }
    }
}

以上代码中main函数的try..catch无法捕获thread中的异常。

那么就只能用try..catch来处理每个线程的异常吗?在java 5中,提供了两种机制来更优雅的处理线程异常。

全局线程异常处理器

程序员可以定义自己的ExceptinHandler并将其设置到jvm上,这样所有的线程异常都交由这个异常处理器处理。

class ExceptionTask implements Runnable {
    @Override
    public void run() {
        System.out.println("Exception Task ");
        throw new RuntimeException("Thread Exception");
    }
}
class CustomizeExceptionHandler implements Thread.UncaughtExceptionHandler {
    @Override
    public void uncaughtException(Thread t, Throwable e) {
        System.out.println("catch thread exception:" + e);
    }
}

public class ExceptionHandleTest {
    public static void main(String[] args) {
        //全局异常
        Thread.setDefaultUncaughtExceptionHandler(new CustomizeExceptionHandler());
        thread.start();
        Thread thread2 = new Thread(new ExceptionTask());
        thread2.start();
    }
}

输出结果:

Exception Task 
catch thread exception:java.lang.RuntimeException: Thread Exception
Exception Task 
catch thread exception:java.lang.RuntimeException: Thread Exception

这样,所有的线程异常都会被CustomizeExceptionHandler处理。

线程私有异常处理器

如果每个线程异常的处理策略不同,那么,也可以将异常处理器的适用范围设置到一个具体的线程上。

class ExceptionTask implements Runnable {
    @Override
    public void run() {
        System.out.println("Exception Task ");
        throw new RuntimeException("Thread Exception");
    }
}

class CustomizeExceptionHandler implements Thread.UncaughtExceptionHandler {
    @Override
    public void uncaughtException(Thread t, Throwable e) {
        System.out.println("catch thread exception:" + e);
    }
}
class CustomizeExceptionHandler2 implements Thread.UncaughtExceptionHandler {

    @Override
    public void uncaughtException(Thread t, Throwable e) {
        System.out.println("catch thread exception2:" + e);
    }
}

public class ExceptionHandleTest {
    public static void main(String[] args) {
        Thread thread = new Thread(new ExceptionTask());
        thread.setUncaughtExceptionHandler(new CustomizeExceptionHandler());
        thread.start();

        Thread thread2 = new Thread(new ExceptionTask());
        thread2.setUncaughtExceptionHandler(new CustomizeExceptionHandler2());
        thread2.start();
    }
}

输出结果为:

Exception Task 
catch thread exception:java.lang.RuntimeException: Thread Exception
Exception Task 
catch thread exception2:java.lang.RuntimeException: Thread Exception

这样,每个线程就可以适用自己的异常处理器来处理各自的异常。

在线程池中使用异常处理器

在jdk1.5以后,推荐的使用线程的方式是线程池,那么,在线程池环境下如何设置线程的异常处理器呢?

由于线程池中的线程都是有ThreadFactory产生的,所以,我们可以实现自己的ThreadFactory,为其产生的每个Thread实例设置一个异常处理器。

另外,也可以像全局线程异常处理器那样设置,但局限性会比较大。

下面看一个如何通过自定义ThreadFactory来设置异常处理器

//自定义ThreadFactory
class CustomizeThreadFactory implements ThreadFactory {
    @Override
    public Thread newThread(Runnable r) {
        Thread thread = new Thread(r);
        //为产生的每个线程实例设置异常处理器
        thread.setUncaughtExceptionHandler(new CustomizeExceptionHandler());
        return  thread;
    }
}

class ExceptionTask implements Runnable {
    @Override
    public void run() {
        System.out.println("Exception Task ");
        throw new RuntimeException("Thread Exception");
    }
}
//自定义异常处理器
class CustomizeExceptionHandler implements Thread.UncaughtExceptionHandler {
    @Override
    public void uncaughtException(Thread t, Throwable e) {
        System.out.println("catch thread exception:" + e);
    }
}

public class ExceptionHandleTest {
    public static void main(String[] args) {
        //为线程池指定线程工厂实例
        ExecutorService executorService = Executors.newCachedThreadPool(new CustomizeThreadFactory());
        executorService.execute(new ExceptionTask());
        executorService.shutdown();
    }
}

输出结果为:

Exception Task 
catch thread exception:java.lang.RuntimeException: Thread Exception

总结一下:

  • 通过实现Thread.UncaughtExceptionHandler接口自定义线程异常处理器
  • 通过Thread.setDefaultUncaughtExceptionHandler()设置全局异常处理器
  • 通过new Thread().setUncaughtExceptionHandler()方法为每个线程实例设置异常处理器
  • 通过实现ThreadFactory接口自定义线程工厂,来为线程池提供带有异常处理功能的线程实例。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值