项目中异常什么时候打印错误和抛出?

3 阅读2分钟

大学生学习记录,如有错误或不足,欢迎指出,感谢!

主要分情况,一般是系统异常或者业务异常。

系统异常(数据库/Redis/网络/空指针等未知错误)

必须:log.error(关键信息,异常 e) + 抛出 自定义系统异常

  • 一定要把异常堆栈 e 打出来,不然上线出问题查不到原因
  • 抛一个有名字的异常(如SystemException),方便全局识别

系统异常写法

@Service
public class OrderService {
    private static final Logger log = LoggerFactory.getLogger(OrderService.class);

    public Order getOrder(Long orderId) {
        try {
            // 数据库操作,可能抛SQL异常/空指针
            return orderMapper.selectById(orderId);
        } catch (Exception e) {
            // ✅ 必须:打印错误日志 + 带上完整异常堆栈e
            log.error("查询订单异常,orderId:{}", orderId, e);
            // ✅ 然后抛一个有名字的自定义异常
            throw new SystemException("查询订单失败,请稍后再试");
        }
    }
}

业务异常(参数错误/订单不存在/库存不足等预期内错误)

只抛自定义业务异常,不打 log.error

  • 业务异常是正常流程里的判断,不是bug,不用打堆栈
  • 全局异常处理器里打个 log.warn提示就行,避免日志泛滥。
public Order createOrder(Order order) {
    // 业务判断,预期内的错误
    if (order.getUserId() == null) {
        // ✅ 只抛业务异常,不打log.errorthrow new BizException("用户ID不能为空");
    }
    if (order.getProductCount() <= 0) {
        throw new BizException("商品数量必须大于0");
    }
    return orderMapper.insert(order);
}

为什么业务异常不打log.error?

  • 业务异常是正常逻辑,不是程序bug
  • 如果你每个校验都打 error, 日志文件会瞬间爆炸
  • 全局异常处理器打个warn就够了

全局异常处理类写法

一般controller 不捕获异常,交给这里的带有@RestControllerAdvice注解异常处理类。

// 全局异常处理类(整个项目只写这1次)
@RestControllerAdvice
public class GlobalExceptionHandler {

    // 捕获【业务异常】
    @ExceptionHandler(BizException.class)
    public Result<Void> handleBizException(BizException e) {
        // 日志只在这里打一次
        log.warn("业务异常:{}", e.getMessage());
        // 统一返回前端
        return Result.fail(e.getMessage());
    }

    // 捕获【所有系统异常】
    @ExceptionHandler(Exception.class)
    public Result<Void> handleException(Exception e) {
        log.error("系统异常", e);
        return Result.fail("服务器繁忙,请稍后再试");
    }
}

注意的细节点

  1. log.error 必须带上 e

java

运行

// 错误(没有堆栈,上线等于没日志)
log.error("查询订单失败");

// 正确(带完整异常堆栈,能定位代码行)
log.error("查询订单失败,orderId:{}", orderId, e);
  1. **抛的异常必须 “有名字”**不要抛ExceptionRuntimeException,要抛自定义的:
  • SystemException(系统错误)
  • BizException(业务错误)全局处理器才能分开处理,返回不同提示。