Java异常处理机制

Posted yaofrankie

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java异常处理机制相关的知识,希望对你有一定的参考价值。

一. 异常类型

1. Exception

  Exception主要分为两种:Runtime Exception、Checked(Compile) Exception

  常见的Runtime Exception,有:NullPointerException、ArithmeticException...

  常见的Checked(Compile) Exception,有:IOException、FileNotFoundException...

  所谓Checked Exception就是在编译期间,你必须对某条、或多条语句进行异常处理,如: try...catch...、throws语句。

  下面介绍下Checked Exception的优缺点:

  • 特点与优点: Java专有,体现Java的设计哲学,没有完善错误处理的代码根本就不会给你机会去执行。
  • 缺点:
    • 必须显式捕捉并处理异常,或显式声明抛出异常,增加程序复杂度
    • 若显式抛出异常,则会增加方法签名与异常的耦合度

2. Error

  Error主要表示一些虚拟机内部错误,如:动态链接失败。

二. 异常处理规则

  • 程序可读性:避免过度使用异常处理代码,减少方法签名与异常的耦合度。
  • 异常原始性:捕获并保留原始异常信息
  • 异常针对性:根据业务需求决定如何处理异常,比如:
    • 当你检查商品库存时发生异常,此时就应终止此次调用,并告诉上层用户详细、明确的原因。
    • 当你获取用户头像失败时,因为该操作不影响整体订单、支付流程,所以不需要终止此次调用,可与上层用户协商处理,比如:返回一个空字符串。

三. 相关问题

  • throw与throws区别?
    • 位置:
      • throws位于方法签名
      • throw位于函数体内
    • 语法格式
      • throws后面跟的是异常类,且一次可以跟多个,只需要以逗号分隔。
      • throw后面跟着的是异常实例,且一次只能跟一个。
    • 命中率
      • throws只是做个防守,并不会真正执行。
      • 一旦执行到throw语句,必定抛出异常。

  • 为什么要有异常处理机制?
    • 无法穷举所有的异常情况。
    • 若异常处理的代码过多,会导致程序可读性变差。

  • 为什么要把原始异常封装一层?
    • 安全性,防止恶意用户获得系统内部信息。
    • 对上层用户更加友好,让其更加明确、详细的知道异常原因。

  • 为什么有那么多类需要实现Closeable或AutoCloseable接口?
    • Java9增强了自动关闭资源的try语句。
    • public class ExceptionTest 
          public static void readFile()
              try(BufferedReader bufferedReader = new BufferedReader(new FileReader("justForTest.txt")))
              
                  System.out.println(bufferedReader.readLine());
               catch (IOException e) 
                  e.printStackTrace();
              
          
      
      
      @Test
      public void readFileTest()
          ExceptionTest.readFile(); // Just for test.
      

      技术图片

四. 实践(自定义RuntimeException)

 

@Getter
public class ServiceException extends RuntimeException 

    private HttpStatus status;
    private ResultCode resultCode;
    private Object     errorData;

    private ServiceException(HttpStatus status, ResultCode resultCode, Object errorData)
        this.status     = status;
        this.resultCode = resultCode;
        this.errorData  = errorData;
    

    public static ServiceException badRequest(ResultCode resultCode, Object errorData)
        return new ServiceException(HttpStatus.BAD_REQUEST, resultCode, errorData);
    


@Getter
public enum ResultCode 
    // errorCode
    SUCCESS(0, "SUCCESS"),
    INVALID_PARAMETER(600, "invalid parameter");

    private final int errorCode;
    private final String errorData;

    ResultCode(int errorCode, String errorData)
        this.errorCode = errorCode;
        this.errorData = errorData;
    


@ControllerAdvice
public class GlobalErrorHandler 

    @ExceptionHandler(ServiceException.class)
    public ResponseEntity<ErrorResponse> handleServiceException(ServiceException ex)
        return ResponseEntity
                .status(ex.getStatus())
                .body(ErrorResponse.failed(ex.getResultCode(), ex.getErrorData()));
    



@ApiModel
@Getter
public class ErrorResponse<T> implements Serializable 

    private static final long serialVersionUID = -2254339918462802230L;

    private final int errorCode;
    private final String errorMsg;
    private final T errorData;

    private ErrorResponse(ResultCode resultCode, T errorData) 
        this.errorCode = resultCode.getErrorCode();
        this.errorMsg  = resultCode.getErrorData();
        this.errorData = errorData;
    

    public static <T> ErrorResponse<T> failed(ResultCode resultCode, T data)
        return new ErrorResponse(resultCode, data);
    


@RestController
public class OrderController 

    @GetMapping(value = "/v1/orders/order_id"/*, produces = "application/toString", "application/json"*/)
    public Order getOrder(@PathVariable("order_id") @NotBlank String orderId)

        Order order = new Order();
        BigDecimal total = new BigDecimal(-1.00, new MathContext(2, RoundingMode.HALF_UP));

        if (total.compareTo(BigDecimal.ZERO) <= 0)
            throw ServiceException.badRequest(ResultCode.INVALID_PARAMETER,
                    "Total is less than zero!");
        
        order.setOrderId(orderId);
        order.setTotal(total);
        return order;
    

 

技术图片

 

五. 参考

  • 疯狂Java讲义(第十章 - 异常处理)
  • JAVA核心知识点整理

以上是关于Java异常处理机制的主要内容,如果未能解决你的问题,请参考以下文章

Java中的异常处理机制的简单原理和应用。

java异常处理详解!!

Java中的异常处理机制

Java中的异常处理机制

java异常处理的机制有哪几种

Java异常处理机制:try...catch...的执行流程