SpringBoot 异常处理的最佳实践

这篇具有很好参考价值的文章主要介绍了SpringBoot 异常处理的最佳实践。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

SpringBoot 异常处理的最佳实践

在 Web 开发中,异常处理是非常重要的一环。在 SpringBoot 框架中,异常处理方式有很多种,但是如何选择最佳实践呢?本文将介绍 SpringBoot 异常处理的最佳实践,并附带代码示例。

SpringBoot 异常处理的最佳实践

异常处理的最佳实践

异常处理的最佳实践包括以下几个方面:

  1. 统一异常处理

在 SpringBoot 中,我们可以使用 @ControllerAdvice 注解实现全局异常处理。使用 @ControllerAdvice 注解,我们可以集中处理所有 Controller 层抛出的异常,而不必在每个 Controller 中单独处理。在 GlobalExceptionHandler 类中定义异常处理方法,并根据具体的业务需求对异常进行相应的处理,可以大大提高代码的可维护性和可读性。

  1. 自定义异常

在 SpringBoot 中,我们可以通过自定义异常类来处理特定的异常情况,并在 GlobalExceptionHandler 类中定义对应的异常处理方法。自定义异常类可以继承 RuntimeException 或 Exception 类,根据具体的业务需求进行选择。自定义异常类可以包含异常码、异常信息等属性,方便我们对异常进行更加细致的处理。

  1. 统一返回结果

在 SpringBoot 中,我们可以定义一个统一的返回结果类,用于封装接口返回结果。在接口返回时,我们可以使用该类作为返回值,并根据具体的业务需求封装返回结果的状态码、消息和数据等信息。

  1. 参数校验

在 SpringBoot 中,我们可以使用 javax.validation.constraints 包中的注解对参数进行校验。在 Controller 层中,我们可以使用 @Validated 注解对参数进行校验,并在对应的 Service 层方法中进行处理。如果参数校验失败,会抛出 MethodArgumentNotValidException 异常,在 GlobalExceptionHandler 类中的 handleMethodArgumentNotValidException 方法中进行处理。

  1. 日志记录

在 SpringBoot 中,我们可以使用日志框架记录系统运行时的异常信息。在 GlobalExceptionHandler 类中的异常处理方法中,我们可以使用日志框架记录异常发生的时间、异常类型、异常信息等相关信息,方便我们进行排查和问题定位。

完整代码示例

下面是一个完整的使用 SpringBoot 异常处理的示例代码:

@ControllerAdvice
public class GlobalExceptionHandler {
    private static final Logger logger = LoggerFactory.getLogger(GlobalExceptionHandler.class);

    @ExceptionHandler(Exception.class)
    @ResponseBody
    public ErrorResponse handleException(Exception e) {
        logger.error("系统异常", e);
        return new ErrorResponse(HttpStatus.INTERNAL_SERVER_ERROR.value(), "系统异常");
    }

    @ExceptionHandler(UserNotFoundException.class)
    @ResponseBody
    public ErrorResponse handleUserNotFoundException(UserNotFoundException e) {
        logger.warn("用户不存在", e);
        return new ErrorResponse(HttpStatus.NOT_FOUND.value(), "用户不存在");
    }

    @ExceptionHandler(MethodArgumentNotValidException.class)
    @ResponseBody
    public ErrorResponse handleMethodArgumentNotValidException(MethodArgumentNotValidException e) {
        BindingResult bindingResult = e.getBindingResult();
        List<ObjectError> allErrors = bindingResult.getAllErrors();
        List<String> errorMessages = new ArrayList<>();
        for (ObjectError error : allErrors) {
            errorMessages.add(error.getDefaultMessage());
        }
        String errorMessage = String.join(",", errorMessages);
        logger.warn("参数无效", e);
        return new ErrorResponse(HttpStatus.BAD_REQUEST.value(), errorMessage);
    }
}

@RestController
@RequestMapping("/user")
public class UserController {
    @Autowired
    private UserService userService;

    @PostMapping("/add")
    public Result addUser(@Validated @RequestBody User user) {
        userService.addUser(user);
        return Result.success();
    }

    @GetMapping("/{id}")
    public Result getUserById(@PathVariable("id") Long id) {
        User user = userService.getUserById(id);
        if (user == null) {
            throw new UserNotFoundException();
        }
        return Result.success(user);
    }
}

@Service
public class UserService {
    @Autowired
    private UserDao userDao;

    public void addUser(User user) {
        userDao.save(user);
    }

    public User getUserById(Long id) {
        return userDao.findById(id).orElse(null);
    }
}

@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {
    @NotNull(message = "用户名不能为空")
    privateLong id;

    @NotBlank(message = "用户名不能为空")
    private String username;

    @NotBlank(message = "密码不能为空")
    private String password;
}

@Data
@NoArgsConstructor
@AllArgsConstructor
public class Result<T> {
    private Integer code;
    private String message;
    private T data;

    public static <T> Result<T> success() {
        return new Result<>(HttpStatus.OK.value(), "成功", null);
    }

    public static <T> Result<T> success(T data) {
        return new Result<>(HttpStatus.OK.value(), "成功", data);
    }

    public static <T> Result<T> error(Integer code, String message) {
        return new Result<>(code, message, null);
    }
}

@Data
@NoArgsConstructor
@AllArgsConstructor
public class ErrorResponse {
    private Integer code;
    private String message;
}

public class UserNotFoundException extends RuntimeException {
}

@Repository
public interface UserDao extends JpaRepository<User, Long> {
}

在上面的代码示例中,我们定义了一个 GlobalExceptionHandler 类,用于处理全局异常。在该类中,我们定义了三个异常处理方法:handleException、handleUserNotFoundException 和 handleMethodArgumentNotValidException。分别用于处理系统异常、用户不存在异常和参数无效异常。在每个方法中,我们都使用日志框架记录异常信息,并返回封装好的 ErrorResponse 对象。

在 UserController 类中,我们使用 @Validated 注解对 addUser 方法中的 User 参数进行了校验。如果校验失败,会抛出 MethodArgumentNotValidException 异常,在 GlobalExceptionHandler 类中的 handleMethodArgumentNotValidException 方法中进行处理。

在 UserService 类中,我们使用了 JpaRepository 接口,对 User 数据进行了持久化操作。

最后,我们定义了 User、Result、ErrorResponse 和 UserNotFoundException 四个类,分别用于封装用户信息、接口返回结果、错误信息和用户不存在异常。

结论

在本文中,我们介绍了 SpringBoot 异常处理的最佳实践,并附带了完整的代码示例。在使用 SpringBoot 进行 Web 开发时,我们应该充分考虑异常处理的方方面面,以保证代码的可维护性和可读性。通过使用统一异常处理、自定义异常、统一返回结果、参数校验和日志记录等技术手段,我们可以更加高效地进行异常处理,提高系统的稳定性和可靠性。文章来源地址https://www.toymoban.com/news/detail-497375.html

到了这里,关于SpringBoot 异常处理的最佳实践的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Java异常处理的20个最佳实践:告别系统崩溃

    在Java编程中,异常处理是一个至关重要的环节,它不仅涉及到程序的稳定性和安全性,还关系到用户体验和系统资源的合理利用。合理的异常处理能够使得程序在面对不可预知错误时,能够优雅地恢复或者给出明确的反馈,而不是简单地崩溃退出。 文章开始前,我们先看下

    2024年02月22日
    浏览(35)
  • Java 异常处理与正则表达式详解,实例演练及最佳实践

    在 Java 代码执行期间,可能会发生各种错误,包括程序员编码错误、用户输入错误以及其他不可预料的状况。 当错误发生时,Java 通常会停止并生成错误消息,这个过程称为抛出异常。 try 语句允许您定义一段代码块,并在其中测试是否发生错误。 catch 语句允许您定义一段代

    2024年03月13日
    浏览(27)
  • JAVA OOM异常可观测最佳实践

    堆溢出-java.lang.OutOfMemoryError: Java heap space。 栈溢出-java.lang.OutOfMemorryError。 栈溢出-java.lang.StackOverFlowError。 元信息溢出-java.lang.OutOfMemoryError: Metaspace。 直接内存溢出-java.lang.OutOfMemoryError: Direct buffer memory。 GC超限-java.lang.OutOfMemoryError: GC overhead limit exceeded。 垃圾回收器就是内存

    2024年02月06日
    浏览(35)
  • SpringBoot 实战 开发中 16 条最佳实践

    Spring Boot是最流行的用于开发微服务的Java框架。在本文中,我将与你分享自2016年以来我在专业开发中使用Spring Boot所采用的最佳实践。这些内容是基于我的个人经验和一些熟知的Spring Boot专家的文章。 在本文中,我将重点介绍Spring Boot特有的实践(大多数时候,也适用于Spri

    2024年02月07日
    浏览(24)
  • Springboot整合mqtt最新教程,完整教程,最佳实践

    前言:   关于整合mqtt网上的教程很多,但大部分都是cv来cv去,中间的监听代码也没有讲清楚。教程也是很久之前的。所以决定自己来写一个教程。废话不多说直接开始教程。 本文只有教程,没有其他废话,如果需要请留言,后续更新下一版(包括主题消息的订阅方式改变

    2024年04月26日
    浏览(21)
  • 【SpringBoot系列】Spring Boot 3核心技术与最佳实践

    强烈推荐 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站: 人工智能 引言 Spring Boot 3 是对 Spring Boot 框架的一个重要更新版本,它延续了 Spring Boot 简化 Spring 应用程序开发的宗旨,进一步提升了开发者体验和应用程

    2024年03月09日
    浏览(53)
  • Ansible最佳实践之Playbook使用过滤器处理网络地址

    使用过滤器检查、验证和操作包含网络信息的变量 理解不足小伙伴帮忙指正 傍晚时分,你坐在屋檐下,看着天慢慢地黑下去,心里寂寞而凄凉,感到自己的生命被剥夺了。当时我是个年轻人,但我害怕这样生活下去,衰老下去。在我看来,这是比死亡更可怕的事。--------王小

    2024年02月15日
    浏览(46)
  • 加解密在开源SpringBoot/SpringCloud微服务框架的最佳实践

    前期内容导读: Java开源RSA/AES/SHA1/PGP/SM2/SM3/SM4加密算法介绍 Java开源AES/SM4/3DES对称加密算法介绍及其实现 Java开源AES/SM4/3DES对称加密算法的验证说明 Java开源RSA/SM2非对称加密算法对比介绍 Java开源RSA非对称加密算法实现 Java开源SM2非对称加密算法实现 Java开源接口微服务代码框架

    2024年02月09日
    浏览(33)
  • Java中处理千万级数据的最佳实践:性能优化指南

    在今天的数字化时代,处理大规模数据已经成为许多Java应用程序的核心任务。无论您是构建数据分析工具、实现实时监控系统,还是处理大规模日志文件,性能优化都是确保应用程序能够高效运行的关键因素。本指南将介绍一系列最佳实践,帮助您在处理千万级数据时提高

    2024年02月03日
    浏览(39)
  • OAuth2在开源SpringBoot/SpringCloud微服务框架的最佳实践

    前期内容导读: Java开源RSA/AES/SHA1/PGP/SM2/SM3/SM4加密算法介绍 Java开源AES/SM4/3DES对称加密算法介绍及其实现 Java开源AES/SM4/3DES对称加密算法的验证说明 Java开源RSA/SM2非对称加密算法对比介绍 Java开源RSA非对称加密算法实现 Java开源SM2非对称加密算法实现 Java开源接口微服务代码框架

    2024年02月11日
    浏览(30)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包