Java并发编程示例(八):处理线程的非受检异常

Java语言中,把异常分为两类:

受检异常: 这类异常必须在throws子句中被显式抛出或者在方法内被捕获。例如,IOException异常或ClassNotFoundException异常。
非受检异常: 这类异常不需要显式抛出或捕获。例如,NumberFormatException异常。

当一个受检异常在Thread对象的run()方法中被抛出时,我们必须捕获并处理它,因为run()方法不能抛出异常。而一个非受检异常在Thread对象的run()方法中被抛出时,默认的行为是在控制台打印出堆栈跟踪信息然后退出程序。

幸运的是,Java为我们提供了一种机制,专门用于处理由Thread对象抛出的非受检异常,以避免程序的退出。

在本节,我们用示例来演示这种机制。

知其然

按照下面所示步骤来实现我们的示例。

1.首先,我们需要实现一个用于处理非受检异常的类。这个类必须实现UncaughtExceptionHandler类,实现在该接口中声明的uncaughtException()方法。在本例中,该类名为ExceptionHandler,uncaughtException()方法将异常以及抛出异常的线程信息打印出来。代码如下:

复制代码 代码如下:

public class ExceptionHandler implements Thread.UncaughtExceptionHandler {
    @Override
    public void uncaughtException(Thread t, Throwable e) {
        System.out.printf("An exception has been captured\\n");
        System.out.printf("Thread: %s\n", t.getId());
        System.out.printf("Exception: %s: %s\n", e.getClass().getName(),
                e.getMessage());
        System.out.printf("Stack Trace: \n");
        e.printStackTrace(System.out);
        System.out.printf("Thread status: %s\n", t.getState());
    }
}

2.实现一个可以抛出非受检异常的类,称为Task,实现Runnable接口,实现run()方法,特意编码一段可以产生非受检异常的代码,例如,将字符串转换成数字。代码如下:

复制代码 代码如下:

public class Task implements Runnable {
    @Override
    public void run() {
        int numero = Integer.parseInt("diguage.com");
    }
}

3.创建程序的主类,Main类,然后实现main()方法。代码如下:

复制代码 代码如下:

public class Main {
    public static void main(String[] args) {

4.创建Task对象,并且创建一个Thread对象来执行之。使用setUncaughtExceptionHandler() 方法设置非受检异常的处理类。然后,启动线程。代码如下:

复制代码 代码如下:

Task task = new Task();
Thread thread = new Thread(task);
thread.setUncaughtExceptionHandler(new ExceptionHandler());
thread.start();

5.运行示例,查看结果。

知其所以然

从下面的输出片段可以看出异常执行的结果。异常被抛出,然后被处理类捕获并将异常信息打印到了控制台。

复制代码 代码如下:

An exception has been captured
Thread: 9
Exception: java.lang.NumberFormatException: For input string: "diguage.com"
Stack Trace:
java.lang.NumberFormatException: For input string: "diguage.com"
    at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)
    at java.lang.Integer.parseInt(Integer.java:492)
    at java.lang.Integer.parseInt(Integer.java:527)
    at com.diguage.books.concurrencycookbook.chapter1.recipe8.Task.run(Task.java:13)
    at java.lang.Thread.run(Thread.java:722)
Thread status: RUNNABLE

Process finished with exit code 0

当一个线程抛出一个异常,并且该异常(这里特指非受检异常)没有捕获时,Java虚拟机会检查是否通过相应方法设置非受检异常处理类,如果以已经设置过,则调用uncaughtException()方法,并将线程和异常作为参数传递给方法。

如果没有设置处理类,Java虚拟机就会在控制台将堆栈跟踪信息打印出来,然后退出程序。

永无止境

Thread类还有一个和非受检异常处理相关的方法。这就是静态方法setDefaultUncaughtExceptionHandler(),该方法可以设置程序中所有线程的非受检异常的处理类。

当线程中抛出一个未捕获的异常时,Java虚拟机会从三个地方寻找异常处理类:

首先,从线程对象中查找异常处理类,这就是我们本节所学内容。如不存在,则从线程所在的线程组(ThreadGroup)中查找异常处理类。关于这部分内容,以后会专门讲解。如果还是不存在,则查找上面刚刚提到的程序默认异常处理类。

如果上面提到的异常处理都不存在,则Java虚拟机将异常的堆栈跟踪信息打印到控制台,然后退出程序。

拿来主义

本文是从 《Java 7 Concurrency Cookbook》 (D瓜哥窃译为 《Java7并发示例集》 )翻译而来,仅作为学习资料使用。没有授权,不得用于任何商业行为。

小有所成

ExceptionHandler类的完整代码

复制代码 代码如下:

package com.diguage.books.concurrencycookbook.chapter1.recipe8;

/**
 * 非受检异常处理类
 * Date: 2013-09-22
 * Time: 23:11
 */
public class ExceptionHandler implements Thread.UncaughtExceptionHandler {
    @Override
    public void uncaughtException(Thread t, Throwable e) {
        System.out.printf("An exception has been captured\n");
        System.out.printf("Thread: %s\n", t.getId());
        System.out.printf("Exception: %s: %s\n", e.getClass().getName(),
                e.getMessage());
        System.out.printf("Stack Trace: \n");
        e.printStackTrace(System.out);
        System.out.printf("Thread status: %s\n", t.getState());
    }
}

Task类的完整代码

复制代码 代码如下:

package com.diguage.books.concurrencycookbook.chapter1.recipe8;

/**
 * 异常生成类
 * Date: 2013-09-22
 * Time: 23:18
 */
public class Task implements Runnable {
    @Override
    public void run() {
        int numero = Integer.parseInt("diguage.com");
    }
}

Main类的完整代码

复制代码 代码如下:

package com.diguage.books.concurrencycookbook.chapter1.recipe8;

/**
 * 示例的主类
 * Date: 2013-09-22
 * Time: 23:20
 */
public class Main {
    public static void main(String[] args) {
        Task task = new Task();
        Thread thread = new Thread(task);
        thread.setUncaughtExceptionHandler(new ExceptionHandler());
        thread.start();
    }
}

时间: 2014-12-04

java进行error捕获和处理示例(java异常捕获)

下面给个小例子,来验证一下error的捕获. 复制代码 代码如下: public class TestCatchError extends Error{ private static final long serialVersionUID = -351488225420878020L; public TestCatchError(){        super();    } public TestCatchError(String msg){        super(msg);    } p

简单的java图片处理类(图片水印 图片缩放)

复制代码 代码如下: import java.awt.AlphaComposite;import java.awt.Color;import java.awt.Font;import java.awt.Graphics;import java.awt.Graphics2D;import java.awt.Image;import java.awt.Toolkit;import java.awt.color.ColorSpace;import java.awt.geom.AffineTransfo

JAVA流控及超流控后的延迟处理实例

本文实例讲述了JAVA流控及超流控后的延迟处理方法.分享给大家供大家参考.具体实现方法如下: 流控检查(每半秒累计,因此最小留空阀值只能做到每秒2条): 复制代码 代码如下: import java.text.SimpleDateFormat; import java.util.Date; import java.lang.Thread;   /**  * 流量控制  *  * @author chenx  */ public class OverflowController {       p

Java进阶教程之异常处理

程序很难做到完美,不免有各种各样的异常.比如程序本身有bug,比如程序打印时打印机没有纸了,比如内存不足.为了解决这些异常,我们需要知道异常发生的原因.对于一些常见的异常,我们还可以提供一定的应对预案.C语言中的异常处理是简单的通过函数返回值来实现的,但返回值代表的含义往往是由惯例决定的.程序员需要查询大量的资料,才可能找到一个模糊的原因.面向对象语言,比如C++, Java, Python往往有更加复杂的异常处理机制.这里讨论Java中的异常处理机制. Java异常处理 异常处理 Java的异

Java基础之java处理ip的工具类

java处理ip的工具类,包括把long类型的Ip转为一般Ip类型.把xx.xx.xx.xx类型的转为long类型.根据掩码位获取掩码.根据 ip/掩码位 计算IP段的起始IP.根据 ip/掩码位 计算IP段的终止IP等方法,可以直接使用! 复制代码 代码如下: package com.hh.test; import java.util.HashMap; import java.util.Map; import org.apache.commons.lang3.StringUtils; /**

java数字图像处理基础使用imageio写图像文件示例

一个BufferedImage的像素数据储存在Raster中,ColorModel里面储存颜色空间,类型等信息,当前Java只支持一下三种图像格式- JPG,PNG,GIF,如何向让Java支持其它格式,首先要 完成Java中的图像读写接口,然后打成jar,加上启动参数- Xbootclasspath/pnewimageformatIO.jar即可. Java中如何读写一个图像文件,使用ImageIO对象即可.读图像文件的代码如下: 复制代码 代码如下: File file = new File

java异常处理机制示例(java抛出异常、捕获、断言)

这是一个介绍基本异常处理的小例子,包括抛出,捕获,断言,日志. Java异常处理通过5个关键字try.catch.throw.throws.finally进行管理.基本过程是用try语句块包住要监视的语句,如果在try语句块内出现异常,则异常会被抛出,你的代码在catch语句块中可以捕获到这个异常并做处理;还有以部分系统生成的异常在Java运行时自动抛出.你也可以通过throws关键字在方法上声明该方法要抛出异常,然后在方法内部通过throw抛出异常对象. 复制代码 代码如下: package

java时间相关处理小结

1.计算某一月份的最大天数 复制代码 代码如下: Calendar time=Calendar.getInstance();time.clear();time.set(Calendar.YEAR,year); //year 为 inttime.set(Calendar.MONTH,i-1);//注意,Calendar对象默认一月为0          int day=time.getActualMaximum(Calendar.DAY_OF_MONTH);//本月份的天数注:在使用set方法之前

Java线程安全问题小结_动力节点Java学院整理

浅谈java内存模型 不同的平台,内存模型是不一样的,但是jvm的内存模型规范是统一的.其实java的多线程并发问题最终都会反映在java的内存模型上,所谓线程安全无非是要控制多个线程对某个资源的有序访问或修改.总结java的内存模型,要解决两个主要的问题:可见性和有序性.我们都知道计算机有高速缓存的存在,处理器并不是每次处理数据都是取内存的.JVM定义了自己的内存模型,屏蔽了底层平台内存管理细节,对于java开发人员,要清楚在jvm内存模型的基础上,如果解决多线程的可见性和有序性. 那么,何谓

分享40个Java多线程问题小结

Java多线程是什么 Java提供的并发(同时.独立)处理多个任务的机制.多个线程共存于同一JVM进程里面,所以共用相同的内存空间,较之多进程,多线程之间的通信更轻量级.依我的理解,Java多线程完全就是为了提高CPU的利用率.Java的线程有4种状态,新建(New).运行(Runnable).阻塞(Blocked).结束(Dead),关键就在于阻塞(Blocked),阻塞意味着等待,阻塞的的线程不参与线程分派器(Thread Scheduler)的时间片分配,自然也就不会使用到CPU.多线程环

Java 集合概览(小结)

Java Collection API提供了一些列的类和接口来帮助我们存储和管理对象集合.其实Java中的集合工作起来像是一个数组,不过集合的大小是可以动态改变的,而且集合也提供了更多高级功能.有了JavaCollectionAPI,我们就不需要自己编写集合类了,大部分Java集合类都位于java.util包里面,还有一些和并发相关的集合类位于java.util.concurrent包中.下面就介绍一下Java API 为我们提供的这些集合类. 一.Java 集合概览 Java中的集合有两大类,

16 个有用的的Java工具类(小结)

在Java中,工具类定义了一组公共方法,这篇文章将介绍Java中使用最频繁及最通用的Java工具类.以下工具类.方法按使用流行度排名,参考数据来源于Github上随机选取的5万个开源项目源码. 一. org.apache.commons.io.IOUtils closeQuietly:关闭一个IO流.socket.或者selector且不抛出异常,通常放在finally块 toString:转换IO流. Uri. byte[]为String copy:IO流数据复制,从输入流写到输出流中,最大支

浅谈常用Java数据库连接池(小结)

概述 在这里所谓的数据库连接是指通过网络协议与数据库服务之间建立的TCP连接.通常,与数据库服务进行通信的网络协议无需由应用程序本身实现,原因有三: 1.实现复杂度大,需要充分理解和掌握相应的通信协议. 2.代码难以复用,每个应用程序都需要独立实现一套对应的网络协议(不同公司之间,同一公司的不同技术栈之间难以复用实现相同协议的代码) 3.性能难以保证,不同的网络协议实现可能存在巨大的性能差距 正因为如此,所以现实的实现方式是: 首先,定义网络协议标准,这样只要支持这个标准协议的数据库就可以使用相

Java Optional实践(小结)

问题描述 在大热的Spring Boot 2.0中,在将原来的泛型改为了Optional,旨在让我们的代码更简洁. 实践 Optional 很简单的一个类,点开它的源代码,其中所有的方法都是与null相关联的. 这是一个简化我们处理null的类. 它就是一个容器,其中有我们想要的对象,但是该对象有时候会是空,所以我们需要使用Optional封装好的方法来获取需要的对象.从而很好地避免了空指针异常. 错误示范 我看到网上很多人这么写: catRepository.findById(id).get(

浅谈JavaScript中小数和大整数的精度丢失

先来看两个问题: 0.1 + 0.2 == 0.3; // false 9999999999999999 == 10000000000000000; // true 第一个问题是小数的精度问题,在业界不少博客里已有讨论.第二个问题,去年公司有个系统的数据库在做数据订正时,发现有部分数据重复的诡异现象.本文将从规范出发,对上面的问题做个小结. 最大整数 JavaScript 中的数字是用 IEEE 754 双精度 64 位浮点数 来存储的,其格式为: s x m x 2^e s 是符号位,表示正负

android 无须root截图方案的实现

通过反射了截取屏 public class EncoderFeeder { public static Bitmap screenshot() { String surfaceClassName; if (VERSION.SDK_INT <= 17) { surfaceClassName = "android.view.Surface"; } else { surfaceClassName = "android.view.SurfaceControl"; }

AngularJS操作键值对象类似java的hashmap(填坑小结)

前言: 我们知道java的hashmap中使用最多的是put(...),get(...)以及remove()方法,那么在angularJS中如何创造(使用)这样一个对象呢 思路分析: 我们知道在java中可以采用链式访问和"[]"访问hashmap的某一个值 具体实现: 链式访问: .factory('ParamsServices', function () { var params = {}; return { get: function (key) { return params.