- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
@ControllerAdvice
注解作用是给Controller控制器添加统一的操作或处理。
对于@ControllerAdvice
,我们比较熟知的用法是结合@ExceptionHandler
用于全局异常的处理(之前总结过一篇全局异常捕获的文章【参数校验 + 全局异常拦截】),但其作用不止于此。ControllerAdvice拆开来就是Controller Advice,关于Advice,在Spring的AOP
中,是用来封装一个切面所有属性的,包括切入点和需要织入的切面逻辑。这里ControllerAdvice也可以这么理解,其抽象级别应该是用于对Controller
进行切面环绕的,而具体的业务织入方式则是通过结合其他的注解来实现的。
ResponseBodyAdvice
可以在注解@ResponseBody
将返回值处理成相应格式之前操作返回值。实现这个接口即可完成相应操作。可用于对response 数据的一些统一封装或者加密等操作
ResponseBodyAdvice
是SpringMVC4.1提供的一个接口,它允许在 执行 @ResponseBody
后自定义返回数据,或者将返回@ResponseEntity
的 Controller Method在写入主体前使用HttpMessageConverter
进行自定义操作。由此可见,它的作用范围为:
@ResponseBody
注解进行标记@ResponseEntity
源码:
package org.springframework.web.servlet.mvc.method.annotation;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.lang.Nullable;
public interface ResponseBodyAdvice<T> {
boolean supports(MethodParameter var1, Class<? extends HttpMessageConverter<?>> var2);
@Nullable
T beforeBodyWrite(@Nullable T var1, MethodParameter var2, MediaType var3, Class<? extends HttpMessageConverter<?>> var4, ServerHttpRequest var5, ServerHttpResponse var6);
}
supports
:该方法返回true时,才会进去下面的 beforeBodyWrite方法。该方法可以添加一些判断条件beforeBodyWrite
:body写入前的操作随便写个实体类
import lombok.Data;
import java.util.List;
@Data
public class UserVo {
private String name;
private Integer age;
private String addr;
private List<String> language;
}
为了返回统一的数据,我们还可以自定义一个返回类ResultVO
import lombok.Data;
import java.io.Serializable;
/**
* @author coderzpw.zhang
* @version 1.0
*/
@Data
public class ResultVO<T> implements Serializable {
// 状态码
private Integer code;
// 提示信息
private String msg;
// 具体内容
private T data;
}
自定义通用枚举类
import lombok.Getter;
/**
* @author coderzpw.zhang
* @version 1.0
*/
@Getter
public enum ResultEnum {
SUCCESS(200, "成功!"),
UN_EXPECTED(500, "系统发生错误,请联系管理员!"),
UN_AUTHORIZED(401, "未认证!"),
NO_PERMISSION(403, "无权限!");
private Integer code;
private String message;
ResultEnum(Integer code, String message) {
this.code = code;
this.message = message;
}
}
然后再写一个生成ResultVo对象的工具类
import com.coderzpw.exception.vo.ResultVO;
import com.coderzpw.exception.enums.ResultEnum;
/**
* @author coderzpw.zhang
* @version 1.0
*/
public class ResultVOUtil {
/**
* 返回成功信息(带返回数据)
* @param object
* @return
*/
public static ResultVO success(Object object) {
ResultVO resultVO = new ResultVO();
resultVO.setData(object);
resultVO.setCode(ResultEnum.SUCCESS.getCode());
resultVO.setMsg(ResultEnum.SUCCESS.getMessage());
return resultVO;
}
/**
* 返回成功信息(不带数据)
* @return
*/
public static ResultVO success() {
return success(null);
}
/**
* 返回错误数据
* @param code
* @param msg
* @return
*/
public static ResultVO error(Integer code, String msg) {
ResultVO resultVO = new ResultVO();
resultVO.setCode(code);
resultVO.setMsg(msg);
return resultVO;
}
/**
* 返回错误数据(枚举类型入参)
* @param resultEnum
* @return
*/
public static ResultVO error(ResultEnum resultEnum) {
ResultVO resultVO = new ResultVO();
resultVO.setCode(resultEnum.getCode());
resultVO.setMsg(resultEnum.getMessage());
return resultVO;
}
}
import com.alibaba.fastjson.JSONObject;
import com.coderzpw.exception.utils.ResultVOUtil;
import com.coderzpw.exception.vo.ResultVO;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;
/**
* @author coderzpw.zhang
* @version 1.0
*/
@ControllerAdvice
public class CommonResultResponseAdvice implements ResponseBodyAdvice<Object> {
@Override
public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
// 直接返回 true,则会处理所有符合条件的返回。如有特殊要求条件,可以在该方法中自行加入拦截逻辑
return true;
}
@Override
public Object beforeBodyWrite(Object o,
MethodParameter methodParameter,
MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass,
ServerHttpRequest serverHttpRequest,
ServerHttpResponse serverHttpResponse) {
// 如果直接响应字符串返回,则会报类型转换异常.
if (o instanceof String) {
return JSONObject.toJSONString(ResultVOUtil.success(o));
}
// 如果已经是ResultVO类型, 就没必要再包装一层了
if (o instanceof ResultVO) {
return o;
}
return ResultVOUtil.success(o);
}
}
import com.coderzpw.exception.vo.UserVo;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* @author coderzpw.zhang
* @version 1.0
*/
@RestController
public class TestController {
@GetMapping("/test1")
public Object test1() {
UserVo userVo = new UserVo();
userVo.setName("coderzpw.zhang");
userVo.setAge(22);
userVo.setAddr("四川省.成都市.高新区");
return userVo;
}
@GetMapping("/test2")
public String test2() {
return "hello";
}
}
我可以定义一个 Spring ControllerAdvice,它由使用自定义注解的 Controller 子集有选择地使用: @RestController @UseAdviceA @Request
这是 web 入口点的主 Controller @Controller @RequestMapping("/webapp") public class WebAppController { @Requ
我有一个 Spring 4 应用程序,其中包含多个用 @Order(someValue) 注释的 ControllerAdvices。此外,我在我的一个外部库中发现了一个 ControllerAdvi
实现前的准备工作 @ControllerAdvice注解的简单介绍 @ControllerAdvice注解作用是给Controller控制器添加统一的操作或处理。 对于@ControllerAdvic
我一直在关注一些有关使用 Spring 框架进行错误处理的教程。我的handleNullPointer方法确实捕获了空指针异常。但是,我想知道该方法之后返回到哪里,因为我想在前端(网页)上显示错误的一
有没有办法从 @ControllerAdvice 获得控制权的地方获取类。 即:如果 PersonController 的执行正在进行,并且我收到一些错误,由于控制权转移到 @ControllerAd
我有一个用 @ControllerAdvice 注释的类,为我的 api 提供了一些中央异常处理。它处理的异常之一是 MethodArgumentNotValidException,当用 @Valid
我想将响应正文设置为异常的错误消息: @ControllerAdvice public class RestResponseEntityExceptionHandler extends Respons
实际上,我在一个单独的项目中使用 @ControllerAdvice 为我的 Sprint 启动项目编写了一个全局 ExceptionHandling,现在我想从另一个项目调用它,但似乎它不起作用,因
我有一个 REST 服务,我需要使用自定义 json 响应发送适当的状态代码,这解释了错误的原因。 我有@Controller类,它公开了一个服务/test,在访问该服务时,我抛出了一个自定义异常(扩
我已经浏览了所有相关的帖子,但我的@ControllerAdvice似乎没有处理从 Controller 类抛出的自定义异常。但是@Controller类中的@ExceptionHandler确实处理
我正在开发一个 Spring MVC/Webflow 应用程序(版本 3.2)并尝试让异常处理工作,我可以在其中将自定义异常消息输出到日志文件和 error.jsp。我遇到的问题是异常处理程序没有被解
我正在为 Spring MVC 中的异常处理开发示例演示应用程序。我正在尝试使用@ControllerAdvice 进行异常处理 我按照 this 中描述的步骤进行操作链接。 但是当我运行我的应用程序
我有以下 Controller 类 package com.java.rest.controllers; @Controller @RequestMapping("/api") public clas
在spring 3.2中,新增了@ControllerAdvice 注解,可以用于定义@ExceptionHandler、@InitBinder、@ModelAttribute,并应用到所有@Req
我有 SimpleMappingExceptionResolver像这样配置。 @Bean(name = "simpleMappingExceptionResolver") public Simple
我不知道为什么,但 @ControllerAdvice覆盖在 Exception 处定义的响应代码级别使用 @ResponseStatus注解。 异常(exception) : @ResponseSt
我制作了一个用于监听异常的通用项目并使用 @ControllerAdvice与 @ExceptionHandler @ControllerAdvice(annotations = RestContro
在我的 REST 服务应用程序中,我计划创建一个 @ControllerAdvice捕获 Controller 抛出的异常并返回 ResponseEntity 的类对象根据错误类型。 但我已经有一个
@RestControllerAdvice 和 @ControllerAdvice 之间的主要区别是什么? 我们是否应该始终将 @RestControllerAdvice 用于休息服务和 @Contr
我是一名优秀的程序员,十分优秀!