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

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

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

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

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

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

系统异常写法

java 复制代码
@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提示就行,避免日志泛滥。
vbnet 复制代码
public Order createOrder(Order order) {
    // 业务判断,预期内的错误
    if (order.getUserId() == null) {
        // ✅ 只抛业务异常,不打log.error!
        throw new BizException("用户ID不能为空");
    }
    if (order.getProductCount() <= 0) {
        throw new BizException("商品数量必须大于0");
    }
    return orderMapper.insert(order);
}

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

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

全局异常处理类写法

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

kotlin 复制代码
// 全局异常处理类(整个项目只写这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

运行

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

// 正确(带完整异常堆栈,能定位代码行)
log.error("查询订单失败,orderId:{}", orderId, e);
  1. **抛的异常必须 "有名字"**不要抛ExceptionRuntimeException,要抛自定义的:
  • SystemException(系统错误)
  • BizException(业务错误)全局处理器才能分开处理,返回不同提示。
相关推荐
该用户已不存在5 小时前
这9款开发工具夯爆了,用了都说好
后端·程序员·全栈
KeepPush5 小时前
Python迭代器与生成器:从原理到实战的深度解析
后端
KeepPush5 小时前
Python itertools 深度指南:用迭代器代数写出更高效的代码
后端
小蜜蜂dry7 小时前
nestjs实战-权限二:角色模块
前端·后端·nestjs
默默且听风7 小时前
Ubuntu 22 环境下 VS Code Codex 插件无法打开的排查与修复记录
后端·ai编程·vibecoding
小蜜蜂dry7 小时前
nestjs实战-权限一: 菜单模块
前端·后端·nestjs
BingoGo7 小时前
PHP 在领域驱动(DDD)设计中的核心实践
后端·php
掘金者阿豪8 小时前
终于!我的第二本书正式出版,吃透 Agentic AI 核心不踩坑
javascript·后端
二月龙8 小时前
Redis 缓存设计避坑指南:穿透、击穿、雪崩与一致性问题
后端
掘金者阿豪8 小时前
运营不会SQL怎么办?我把数据库变成了大家都会用的表格
后端