原文: http://zetcode.com/springboot/exceptionhandler/

Spring Boot @ExceptionHandler教程展示了如何使用 Spring @ExceptionHandler处理异常。

Spring 是流行的 Java 应用框架,而 Spring Boot 是 Spring 的演进,可以帮助轻松地创建独立的,生产级的基于 Spring 的应用。

@ExceptionHandler是用于在特定处理器类或处理器方法中处理异常的注解。 在 Servlet 环境中,我们可以将@ExceptionHandler注解与@ResponseStatus结合起来以定义 HTTP 响应的响应状态。

Spring Boot @ExceptionHandler示例

在以下应用中,我们演示@ExceptionHandler的用法。 主页中的 HTML 链接调用控制器的方法,该方法将返回数据或引发异常。

  1. pom.xml
  2. src
  3. ├── main
  4. ├── java
  5. └── com
  6. └── zetcode
  7. ├── Application.java
  8. ├── controller
  9. └── MyController.java
  10. ├── exception
  11. └── MyDataException.java
  12. └── service
  13. ├── IDataService.java
  14. └── MyDataService.java
  15. └── resources
  16. ├── static
  17. ├── index.html
  18. └── showError.html
  19. └── templates
  20. └── showData.ftl
  21. └── test
  22. └── java

这是 Spring 应用的项目结构。

pom.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
  5. http://maven.apache.org/xsd/maven-4.0.0.xsd">
  6. <modelVersion>4.0.0</modelVersion>
  7. <groupId>com.zetcode</groupId>
  8. <artifactId>springbootexceptionhandlerex</artifactId>
  9. <version>1.0-SNAPSHOT</version>
  10. <packaging>jar</packaging>
  11. <properties>
  12. <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  13. <maven.compiler.source>11</maven.compiler.source>
  14. <maven.compiler.target>11</maven.compiler.target>
  15. </properties>
  16. <parent>
  17. <groupId>org.springframework.boot</groupId>
  18. <artifactId>spring-boot-starter-parent</artifactId>
  19. <version>2.1.1.RELEASE</version>
  20. </parent>
  21. <dependencies>
  22. <dependency>
  23. <groupId>org.springframework.boot</groupId>
  24. <artifactId>spring-boot-starter-web</artifactId>
  25. </dependency>
  26. <dependency>
  27. <groupId>org.springframework.boot</groupId>
  28. <artifactId>spring-boot-starter-freemarker</artifactId>
  29. </dependency>
  30. </dependencies>
  31. <build>
  32. <plugins>
  33. <plugin>
  34. <groupId>org.springframework.boot</groupId>
  35. <artifactId>spring-boot-maven-plugin</artifactId>
  36. </plugin>
  37. </plugins>
  38. </build>
  39. </project>

这是 Maven pom.xml文件。 spring-boot-starter-freemarker是 Freemarker 模板引擎的依赖项; spring-boot-maven-plugin将 Spring 应用打包到可执行的 JAR 或 WAR 归档文件中。

com/zetcode/controller/MyController.java

  1. package com.zetcode.controller;
  2. import com.zetcode.exception.MyDataException;
  3. import com.zetcode.service.IDataService;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.stereotype.Controller;
  6. import org.springframework.web.bind.annotation.ExceptionHandler;
  7. import org.springframework.web.bind.annotation.RequestMapping;
  8. import org.springframework.web.servlet.ModelAndView;
  9. import java.util.HashMap;
  10. import java.util.Map;
  11. @Controller
  12. public class MyController {
  13. @Autowired
  14. private IDataService dataService;
  15. @RequestMapping(value = "/getData")
  16. public ModelAndView getData() {
  17. var data = dataService.findAll();
  18. Map<String, Object> params = new HashMap<>();
  19. params.put("values", data);
  20. return new ModelAndView("showData", params);
  21. }
  22. @ExceptionHandler(MyDataException.class)
  23. public String handleError(MyDataException e) {
  24. return "redirect:/showError.html";
  25. }
  26. }

MyControllergetData()方法调用服务方法,并将检索到的数据存储到列表中。 数据被发送到showData视图。 如果是MyDataException,则控制器将重定向到错误页面。

  1. @ExceptionHandler(MyDataException.class)
  2. public String handleError(MyDataException e) {
  3. return "redirect:/showError.html";
  4. }

handleError()@ExceptionHandler装饰。 MyDataException的处理器已激活。 在方法的主体中,我们重定向到showError.html页面。

com/zetcode/exception/MyDataException.java

  1. package com.zetcode.exception;
  2. public class MyDataException extends RuntimeException {
  3. public MyDataException(String message) {
  4. super(message);
  5. }
  6. }

我们定义一个自定义MyDataException

com/zetcode/service/IDataService.java

  1. package com.zetcode.service;
  2. import java.util.List;
  3. public interface IDataService {
  4. List<String> findAll();
  5. }

IDataService包含契约方法。

com/zetcode/service/MyDataService.java

  1. package com.zetcode.service;
  2. import com.zetcode.exception.MyDataException;
  3. import java.util.ArrayList;
  4. import java.util.List;
  5. import java.util.Random;
  6. import org.springframework.stereotype.Service;
  7. @Service
  8. public class MyDataService implements IDataService {
  9. @Override
  10. public List<String> findAll() {
  11. var r = new Random();
  12. if (r.nextBoolean()) {
  13. throw new MyDataException("Failed to retrieve data");
  14. }
  15. var data = new ArrayList<String>();
  16. data.add("yellow moon");
  17. data.add("brisk pace");
  18. data.add("empty bottle");
  19. data.add("beautiful weather");
  20. return data;
  21. }
  22. }

MyDataService实现IDataServicefindAll()方法。 该方法返回数据或抛出MyDataException

  1. var r = new Random();
  2. if (r.nextBoolean()) {
  3. throw new MyDataException("Failed to retrieve data");
  4. }

findAll()方法随机抛出MyDataException。 然后在控制器中处理异常。

  1. var data = new ArrayList<>();
  2. data.add("yellow moon");
  3. data.add("brisk pace");
  4. data.add("empty bottle");
  5. data.add("beautiful weather");
  6. return data;

如果没有例外,我们将返回一个字符串列表。

resources/static/index.html

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>Home page</title>
  5. <meta charset="UTF-8">
  6. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7. </head>
  8. <body>
  9. <a href="/getData">Get data</a>
  10. </body>
  11. </html>

这是主页。 它包含一个链接,该链接调用我们的控制器方法以获取一些数据。

resources/static/showError.html

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>Error</title>
  5. <meta charset="UTF-8">
  6. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7. </head>
  8. <body>
  9. <p>Failed to retrieve data</p>
  10. </body>
  11. </html>

这是一个错误页面。 抛出MyDataException时显示。

resources/templates/showData.ftl

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>Data</title>
  5. <meta charset="UTF-8">
  6. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7. </head>
  8. <body>
  9. <h2>Data</h2>
  10. <ul>
  11. <#list values as val>
  12. <li>${val}</td>
  13. </#list>
  14. </ul>
  15. </body>
  16. </html>

showData.ftl是一个 Freemarker 模板文件,它在 HTML 列表中显示所有检索到的数据。

com/zetcode/Application.java

  1. package com.zetcode;
  2. import org.springframework.boot.SpringApplication;
  3. import org.springframework.boot.autoconfigure.SpringBootApplication;
  4. @SpringBootApplication
  5. public class Application {
  6. public static void main(String[] args) {
  7. SpringApplication.run(Application.class, args);
  8. }
  9. }

Application是引导 Spring Boot 应用的入口点。

在本教程中,我们展示了如何使用@ExceptionHandler在 Spring 应用中处理异常。 您可能也对相关教程感兴趣: Spring Boot Flash 属性教程Spring Boot @ResponseStatus教程Spring Boot @PathVariable教程Spring Boot @RequestParam教程Java 教程,或显示所有 Spring Boot 教程