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

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

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

系统异常(数据库/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(业务错误)全局处理器才能分开处理,返回不同提示。
相关推荐
啷咯哩咯啷2 小时前
纯本地运行的私人文档知识库
前端·人工智能·后端
阿丰资源2 小时前
基于SpringBoot的房产销售系统设计与实现(附源码+数据库+文档,一键运行)
数据库·spring boot·后端
aLTttY2 小时前
Spring Boot整合AI大模型实现智能问答系统实战
人工智能·spring boot·后端
小江的记录本2 小时前
【微服务与云原生架构】DevOps、CI/CD流水线、GitOps 系统性知识体系
分布式·后端·ci/cd·微服务·云原生·架构·devops
河阿里2 小时前
Spring AOP:企业级实战教学
java·后端·spring
Rust研习社3 小时前
使用 Tonic 构建高性能异步 gRPC 服务
开发语言·网络·后端·http·rust
DevilSeagull3 小时前
Rust 方法语法:从定义到实践
开发语言·后端·rust
每天进步一点_JL3 小时前
Java 线程池深度解析:从零开始理解并发编程的核心工具
后端
每天进步一点_JL4 小时前
Spring 到底在做什么?从零开始理解 Java 企业开发的核心框架
后端·spring