Java异常处理机制

简介: Java异常处理机制一. 异常类型Exception  Exception主要分为两种:Runtime Exception、Checked(Compile) Exception。  常见的Runtime Exception,有:NullPointerException、ArithmeticException.

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

  1. 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 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核心知识点整理
原文地址https://www.cnblogs.com/YaoFrankie/p/11440626.html

相关文章
|
11天前
|
Java 数据库连接 开发者
Java的Shutdown Hook机制:优雅地关闭应用程序
Java的Shutdown Hook机制:优雅地关闭应用程序
18 1
|
13天前
|
Java 程序员 开发者
深入理解Java并发编程:线程同步与锁机制
【4月更文挑战第30天】 在多线程的世界中,确保数据的一致性和线程间的有效通信是至关重要的。本文将深入探讨Java并发编程中的核心概念——线程同步与锁机制。我们将从基本的synchronized关键字开始,逐步过渡到更复杂的ReentrantLock类,并探讨它们如何帮助我们在多线程环境中保持数据完整性和避免常见的并发问题。文章还将通过示例代码,展示这些同步工具在实际开发中的应用,帮助读者构建对Java并发编程深层次的理解。
|
1天前
|
安全 Java 数据安全/隐私保护
Java一分钟之-Java反射机制:动态操作类与对象
【5月更文挑战第12天】本文介绍了Java反射机制的基本用法,包括获取Class对象、创建对象、访问字段和调用方法。同时,讨论了常见的问题和易错点,如忽略访问权限检查、未捕获异常以及性能损耗,并提供了相应的避免策略。理解反射的工作原理和合理使用有助于提升代码灵活性,但需注意其带来的安全风险和性能影响。
13 4
|
2天前
|
Java 数据库连接 数据库
【JAVA基础篇教学】第六篇:Java异常处理
【JAVA基础篇教学】第六篇:Java异常处理
|
3天前
|
存储 Java API
关于Java异常处理的9条原则
关于Java异常处理的9条原则
|
3天前
|
Java 数据安全/隐私保护
java中异常处理机制
java中异常处理机制
11 1
|
4天前
|
算法 安全 Java
深入探索Java中的并发编程:CAS机制的原理与应用
总之,CAS机制是一种用于并发编程的原子操作,它通过比较内存中的值和预期值来实现多线程下的数据同步和互斥,从而提供了高效的并发控制。它在Java中被广泛应用于实现线程安全的数据结构和算法。
19 0
|
4天前
|
Java API 开发者
解密Java反射机制与动态代理
解密Java反射机制与动态代理
9 0
|
4天前
|
IDE Java 测试技术
java异常处理及错误调试技巧
java异常处理及错误调试技巧
9 0
|
6天前
|
Java 数据库连接 开发者
Java中的异常处理机制详解
Java异常处理是确保程序健壮的关键,涉及Throwable的Error和Exception子类。Error由JVM抛出,不建议捕获;Exception分为检查异常(需要捕获)和未检查异常。处理异常的关键字有try、catch、finally、throw和throws。最佳实践包括捕获具体异常、不吞没异常、释放资源和避免滥用异常。示例展示了如何在main方法中处理IOException,并在finally块中进行资源清理。
10 1