Java 改善异常处理的 6 个技巧

Wesley13
• 阅读 674

在软件开发过程中,往往会出现一些不可预知的错误,这些错误有可能花费你数小时甚至数天的时间。异常处理通常是防止这种未知错误的常用措施,它的好处是你不用再绞尽脑汁去考虑各种错误,这为处理某一类错误提供了一个很有效的方法,使编程效率大大提高。

本文将为你介绍6个技巧,来帮助你提高异常处理的能力。

  1. 使用一个单一的、系统范围的异常类

不要针对每种异常类型创建单独的类,而是只创建一个,并使它继承RuntimeException。这可以减少类的数量,并移除你不会去处理的需要声明的异常。

我知道你在想什么:如何告诉异常处理程序,这些是否是同一类型?如何跟踪特定类型的属性?继续读下去。

  1. 为错误代码使用枚举

大多数开发者会把导致异常的原因放到消息中,出现异常时,查看日志文件即可。但是这也有一些缺点:

  • 消息不能被翻译
  • 消息不能很容易地映射为易读的文本
  • 无法从程序方面对消息进行检查

由于每个开发者的语言习惯不一样,同样的错误可能有不同的描述形式。

更好的办法是使用枚举来表示异常的类型。针对每种错误创建一个枚举,并使枚举实现一个ErrorCode接口,然后将它引用为异常中的一个字段。

当抛出异常时,只需在适当的枚举中传递即可。

throw new SystemException(PaymentCode.CREDIT_CARD_EXPIRED);  

现在,当你需要测试具体情况时,只需比较异常代码即可。

catch (SystemException e) {  
    if (e.getErrorCode() == PaymentCode.CREDIT_CARD_EXPIRED) {  
        ...  
    }  
}  

在资源包中使用错误代码作为关键字进行检索,即可取回易读的、国际化的文本。

public class SystemExceptionExample3 {  
   
    public static void main(String[] args) {  
        System.out.println(getUserText(ValidationCode.VALUE_TOO_SHORT));  
    }  
   
    public static String getUserText(ErrorCode errorCode) {  
        if (errorCode == null) {  
            return null;  
        }  
        String key = errorCode.getClass().getSimpleName() + "__" + errorCode;  
        ResourceBundle bundle = ResourceBundle.getBundle("com.northconcepts.exception.example.exceptions");  
        return bundle.getString(key);  
    }  
}  
  1. 在枚举类型中添加错误代码

在某些情况下,一个数字形式的错误代码可以对应一个异常,例如HTTP响应。在这种情况下,在ErrorCode接口中添加一个getNumber方法,并在每个枚举类型中实现它。

public enum PaymentCode implements ErrorCode {  
  SERVICE_TIMEOUT(101),  
  CREDIT_CARD_EXPIRED(102),  
  AMOUNT_TOO_HIGH(103),  
  INSUFFICIENT_FUNDS(104);  

  private final int number;  

  private PaymentCode(int number) {  
    this.number = number;  
  }  

  @Override  
  public int getNumber() {  
    return number;  
  }  
}  

这些数字可以是全局唯一的,或者每个枚举类型只对应一个数字。你甚至可以使用隐式ordinal()方法,或者从一个文件/数据库中加载一些数字。

  1. 将动态字段添加到异常处理中

好的异常处理也意味着记录相关数据,而不仅仅是堆栈跟踪。这样做会节省你大量用于诊断和重现错误的时间。当你的程序停止工作时,也无需客户告诉你,他们做了什么。

做到这一点最简单的方法是在异常处理中增加一个java.util.Map字段。该字段主要作用是保留所有异常相关的数据。如果你使用fluent interface(连贯接口)模式,你还需要添加一个通用的setter方法。

抛出异常,并带有相关的数据,类似于下面的形式:

throw new SystemException(ValidationCode.VALUE_TOO_SHORT)  
  .set("field", field)  
  .set("value", value)  
  .set("min-length", MIN_LENGTH);  
  1. 防止不必要的嵌套

长且多余的堆栈跟踪,对谁都没有好处。更糟的是,它们浪费你的时间和资源。当重新抛出异常时,调用一个静态封装方法,而不是异常的构造函数。封装方法将决定何时嵌套异常,以及何时返回原来的实例。

public static SystemException wrap(Throwable exception, ErrorCode errorCode) {  
  if (exception instanceof SystemException) {  
    SystemException se = (SystemException)exception;  
    if (errorCode != null && errorCode != se.getErrorCode()) {  
      return new SystemException(exception.getMessage(), exception, errorCode);  
    }  
    return se;  
  } else {  
    return new SystemException(exception.getMessage(), exception, errorCode);  
  }  
}  
   
public static SystemException wrap(Throwable exception) {  
  return wrap(exception, null);  
}  

重新抛出异常的代码类似于:

} catch (IOException e) {  
  throw SystemException.wrap(e).set("fileName", fileName);  
}  
  1. 使用一个有Web控制面板的中央记录器

根据你的情况,访问产品日志可能会相当麻烦。因为这可能会涉及到多个中间人。

如果你在一个多服务器环境中,事情会更糟。找到出问题的服务器,并确定该问题只影响这一台服务器,这可能相当令人头痛。 我的建议是:

  • 将日志聚合到某一位置,最好是数据库
  • 允许从Web浏览器访问该数据库

有很多方法可以做到这一点,比如:日志收集器、远程记录器、JMX代理、系统监控软件等,你甚至可以自己构建一个。一旦你拥有它,你将能够:

  • 在几秒钟内解决问题。
  • 每个异常都有一个URL对应。
  • 维护人员无需通过其他人的帮助即可确定异常原因
  • 防止测试人员为同样的错误创建多个tickets。
  • 为企业节省资金
  • 节省你的时间,不会影响你的周末度假 异常处理示例下载:NorthConcepts-Exceptions.zip

对此,你有什么好的建议?欢迎评论。

原文:6 Tips to Improve Your Exception Handling

点赞
收藏
评论区
推荐文章
kenx kenx
3年前
SpringBoot优雅的全局异常处理
前言在日常项目开发中,异常是常见的,但是如何更高效的处理好异常信息,让我们能快速定位到BUG,是很重要的,不仅能够提高我们的开发效率,还能让你代码看上去更舒服,SpringBoot的项目已经有一定的异常处理了,但是对于我们开发者而言可能就不太合适了,因此我们需要对这些异常进行统一的捕获并处理。SpringBoot默认的错误处理机制返回错误页面默认返回W
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
如何优雅的处理异常
Java语言按照错误严重性,从throwale根类衍生出Error和Exception两大派系。本文从异常的定义、处理异常的方式、如何优雅的抛出异常以及处理异常等方面来聊聊如何异常这件事
Wesley13 Wesley13
3年前
java异常处理
_1.异常的分类_Error:称为错误,有java虚拟机生成并抛出,包括动态链接失败、虚拟机错误等,程序对其不做处理。Exception:所以异常类的父类,其子类对应了各种各样可能出现的异常,一般需要用户显示的声明或捕获。RuntimeException:一类特殊的异常,如被0除,数组下标超范围等,其产生比较频繁,处理比较麻烦,如果显示
Wesley13 Wesley13
3年前
03.Android崩溃Crash库之ExceptionHandler分析
目录总结00.异常处理几个常用api01.UncaughtExceptionHandler02.Java线程处理异常分析03.Android中线程处理异常分析04.为何使用setDefaultUncaughtExceptionHandler前沿上一篇整体介绍了crash崩溃
Wesley13 Wesley13
3年前
Java 的Throwable、error、exception的区别
1.  什么是异常?异常本质上是程序上的错误,包括程序逻辑错误和系统错误。比如使用空的引用(NullPointerException)、数组下标越界(IndexOutOfBoundsException)、内存溢出错误等。Throwable类是Java语言中所有错误或异常的超类。有两个重要的子类:Exception(异常)和Error(错误),
Stella981 Stella981
3年前
Exception、Thorow、Throws、TryCatch
一、异常 概述:  异常指的是不正常,指的是程序中出现了某种问题       java中,所有问题都可以使用一个类来表示,这个类叫做Throwable  Throwable:       Throwawble是java中所有异常和错误的父类      Error: 表示错误,指的是不可挽回的严重问题    
Stella981 Stella981
3年前
PlayJava Day020
1.异常Exception补充:①错误(Error)指的是致命性错误,一般无法处理②异常以类的形式封装程序可以处理的异常对应的类是java.lang.Exception及其子类运行时异常对应的类是java.lang.RuntimeException错误异常对应的类是java.lang.Error③异常相关类的继承树:java.la
小万哥 小万哥
1年前
C++异常和错误处理机制:如何使您的程序更加稳定和可靠
在C编程中,异常处理和错误处理机制是非常重要的。它们可以帮助程序员有效地处理运行时错误和异常情况。本文将介绍C中的异常处理和错误处理机制。什么是异常处理?异常处理是指在程序执行过程中发生异常或错误时,程序能够捕获并处理这些异常或错误的机制。例如,当
小万哥 小万哥
6个月前
C++ 异常处理机制详解:轻松掌握异常处理技巧
C异常处理C异常处理机制允许程序在运行时处理错误或意外情况。它提供了捕获和处理错误的一种结构化方式,使程序更加健壮和可靠。异常处理的基本概念:异常:程序在运行时发生的错误或意外情况。抛出异常:使用throw关键字将异常传递给调用堆栈。捕获异常:使用