应依问题性质选择:异常用于意外错误,返回值用于可预期业务分支;资源不可用、违反契约、系统故障必须用异常,查找未命中等常见失败宜用返回值。
Java中该用异常还是返回值,关键看问题性质:异常用于处理“意外的、不该发生”的错误;返回值适合“可预期的、业务逻辑中的正常分支”。选错会降低代码可读性,甚至掩盖真正问题。
当操作失败表示程序处于不合法状态,或调用方无法合理恢复时,必须抛异常:
FileNotFoundException)、网络连接超时(IOException)IllegalArgumentException)、空指针(NullPointerException)OutOfMemoryError)、类加载失败(NoClassDefFoundError)当失败是业务流程中常见、可预测且调用方能主动应对时,返回值更清晰:
null或Optional.empty(),比抛UserNotFoundException更自然DateTimeFormatter.parseBest()返回null或封装结果对象,避免把格式错误全变成异常PaymentResult.failure("余额不足"),而非抛InsufficientBalanceException
这些做法看似方便,实则破坏错误处理语义:
立即学习“Java免费学习笔记(深入)”;
try-catch代替if (map.containsKey(key))——性能差,且混淆错误与逻辑IOException后静默返回-1,调用方完全不知底层失败ValidationException却不带字段名和错误值,排查困难catch,却对大多数场景毫无意义(如SQLException在DAO层已封装为运行时异常更合理)真实项目中不必非此即彼,可以分层协作:
异常:IO、数据库、HTTP客户端等直接抛有意义的运行时异常(如HttpClientException)OrderService.placeOrder()返回Result
@ExceptionHandler把各类异常转为标准JSON错误体,对前端屏蔽实现细节findUserById(id)返回Optional,比返回null或抛异常都更安全、明确