统一处理controller层接口返回的数据

简介: 要对controller层的内容进行统一返回,需要用到 @ControllerAdvice ResponseBodyAdvice


@RestControllerAdvice和ResponseBodyAdvice统一处理controller层接口返回
 

  1. 理论知识

 

要对controller层的内容进行统一返回,需要用到 @ControllerAdvice ResponseBodyAdvice

@RestControllerAdvice
一个方便的注释,它本身带有@ControllerAdvice和@ResponseBody注释,携带此注释的类型被视为控制器通知 总之,它就是对controller层的方法加强

ResponseBodyAdvice
允许在@ResponseBody或ResponseEntity控制器方法执行之后,但在使用HttpMessageConverter编写body之前定制响应

简单理解:ResponseBodyAdvice接口是在controller层方法执行之后,在response返回给前端数据之前对reponse的数据进行处理,可以对数据进行统一的处理,从而可以使返回数据格式一致。

  1. 举例说明

2.1 编写统一返回数据格式代码
@RestControllerAdvice
@Slf4j
public class UnifiedAdvice implements ResponseBodyAdvice {

@Override
public boolean supports(MethodParameter returnType, Class converterType) {
    //若加了@ResponseNotIntercept 则该方法不用做统一的拦截
    AnnotatedElement annotatedElement = returnType.getParameterType();
    ResponseNotIntercept annotation = AnnotationUtils.findAnnotation(annotatedElement, ResponseNotIntercept.class);
    return annotation == null;
}

@Override
public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
    if (body instanceof CommonResult) return body;
    CommonResult<Object> objectCommonResult = new CommonResult<>(ResultCode.SUCCESS, body);
    //若未封装 则对其进行封装
    return objectCommonResult;
}

}
复制代码
说明:实现ResponseBodyAdvice接口 需要重写supports,beforeBodyWrite方法;

supports:是否支持给定的控制器方法返回类型和选定的HttpMessageConverter类型; 若不支持 则就不会对数据进行做统一处理,就像上面代码,若加了@ResponseNotIntercept注解,则不会进行拦截(@ResponseNotIntercept是自己自定义的一个注解)

参数:returnType:返回类型; converterType:选择的转换器类型

返回:若返回结果为true,则调用beforeBodyWrite

beforeBodyWrite:在选择HttpMessageConverter之后以及在调用其write方法之前调用。

参数:body:你传入的数据;returnType:controller层方法返回的类型;selectedContentType :通过内容协商选择的内容类型;selectedConverterType:选择要写入响应的转换器类型;request/reponse:当前请求和响应

返回:传入的数据或修改的(可能是新的)实例

2.2编写contrller层方法
@RestController
@Slf4j
@RequestMapping("/baseInfo")
public class BaseInfoController {

@Autowired
private BaseInfoService baseInfoService;
/**
 * 添加基本信息
 * @param info
 * @return
 */
@PostMapping("/addBaseInfo")
public BaseInfo addBaseInfo(@RequestBody BaseInfo info){
    BaseInfo baseInfo = baseInfoService.addBaseInfo(info);
    return baseInfo;
}

}
复制代码
控制层返回的是一个对象,业务层 数据层方法省略 对只想执行后返回的结果做了统一的处理:

2.3统一返回对象
@Data
@Slf4j
public class CommonResult {

private String code;
private String message;
private T Data;

public CommonResult() {
}

public CommonResult(T data) {
    Data = data;
}

/**
 * 若只传入code码 默认传入的数据为null
 * @param rc
 */
public CommonResult(ResultCode rc) {
    this(rc, null);
}
public CommonResult(ResultCode rc, T data) {
    this.code  = rc.getCode();
    this.message = rc.getMsg();
    this.Data = data;
}


public static<T> CommonResult<T> errorResult(ResultCode rc,T data){
    CommonResult<T> commonResult = new CommonResult<>();
    commonResult.code = rc.getCode();
    commonResult.message = rc.getMsg();
    commonResult.Data = data == null?(T) "" :data;
    log.error("{}",commonResult);
    return commonResult;
}

}
复制代码
ResultCode是自己自定的一些枚举类 例如部分:

  1. 常见错误分析

但是就如上面这些就完全正确了吗?其实不然 若在controller层方法返回字符串会出现什么情况?请看:

@PostMapping("/test")
public String addBaseInfo(){

return "我返回的是一个字符串";

}
复制代码

控制台报错:

org.springframework.web.util.NestedServletException: Request processing failed; nested exception is java.lang.ClassCastException: com.flscode.publicApi.Response.CommonResult cannot be cast to java.lang.String

Caused by: java.lang.ClassCastException: com.flscode.publicApi.Response.CommonResult cannot be cast to java.lang.String

解决

分析: 正常数据返回:

字符串数据返回 打断点:

原因: SpringMVC 默认会注册一些自带的HttpMessageConvertor(从先后顺序排列分别为ByteArrayHttpMessageConverter、StringHttpMessageConverter、ResourceHttpMessageConverter,SourceHttpMessageConverter、AllEncompassingFormHttpMessageConverter),后端服务使用Restful API的形式,前后端得规范一般是json格式,SpringMVC 自带MappingJackson2HttpMessageConverter,在依赖中引入 jackson 包后,容器会把MappingJackson2HttpMessageConverter自动注册到 converter 链的末尾 (这端话来自:blog.csdn.net/weixin_4433…

因此 从上面两张图对比可以看出,此处得方法是要去循环遍历HttpMessageConverter集合,如果对应得转换器能够使用 则会使用该转换器,当你返回得数据是字符串时,因为StringHttpMessageConverter会先被遍历到,这时会认为StringHttpMessageConverter可以使用,因此在ResponseBodyAdvice封装数据时就会报错

修改后:

@Override
public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {

if (body instanceof CommonResult) return body;
CommonResult<Object> objectCommonResult = new CommonResult<>(ResultCode.SUCCESS, body);
if (body instanceof String){
    String s = JSON.toJSONString(new CommonResult<>(ResultCode.SUCCESS, body));
    return s;
}
//若未封装 则对其进行封装
return objectCommonResult;

}
复制代码
controller层:

@PostMapping(value = "/test",produces = "application/json; charset=UTF-8")
public String test(){

return "我返回的是一个字符串";

}
复制代码
返回结果:

  1. 总结

我们在前后端分离开发中,后端一般会返回一个统一格式给前端,若每在一个controller层写一个方法,就要封装一下CommonResult,这样就写了许多不必要得代码,因此,就在controller层每写一个方法,就让他返回它相应得数据即可,不用每次都去封装CommonResult对象,我们将封装CommonResult对象做统一拦截:利用 @RestControllerAdvice和 ResponseBodyAdvice做统一处理,在这个过程中要注意方法返回字符串要做相应得处理,原因可以参考第3点。

作者:又菜又想玩的XXX
链接:https://juejin.cn/post/6991635395758784520
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

相关文章
|
3月前
|
JSON 前端开发 Java
如何封装接口返回结构?
本文详细探讨了API接口返回结构统一化的必要性及其带来的优势,如降低开发的心智负担、减少前端开发难度和提高代码可维护性等。同时也分析了其潜在的缺点,例如灵活性降低和开发成本增加等问题。文章进一步讨论了在Spring Boot中实现统一接口返回结构的具体方法和技术细节,包括如何处理HTTP状态码、返回单个字符串的情况以及如何封装无返回值的接口等。此外,还介绍了如何利用Spring Boot的`ResponseBodyAdvice`和`@RestControllerAdvice`等特性来自动包装控制器方法的返回值及异常处理,以达到更加一致和标准化的接口响应结构。
如何封装接口返回结构?
|
3月前
|
Java 应用服务中间件 HSF
Java应用结构规范问题之dal层中的mapper数据源类型进行组织的问题如何解决
Java应用结构规范问题之dal层中的mapper数据源类型进行组织的问题如何解决
|
3月前
|
设计模式 Java 测试技术
分层设计:Service 层真的需要实现接口吗?
【8月更文挑战第4天】在软件开发领域,分层设计是一种广泛应用且高效的设计模式,它通过将系统划分为不同的逻辑层(如表现层、服务层、数据访问层等),来提高代码的可维护性、可扩展性和可测试性。其中,Service层作为业务逻辑处理的核心,其设计尤为重要。那么,Service层是否真的需要实现接口呢?这个问题值得我们深入探讨。
152 8
|
3月前
|
前端开发 安全 小程序
如何设计 API 接口,实现统一格式返回?
如何设计 API 接口,实现统一格式返回?
48 0
|
5月前
|
设计模式 JSON 前端开发
SSMP整合案例第四步 表现层controller开发及用Result进行统一消息处理
SSMP整合案例第四步 表现层controller开发及用Result进行统一消息处理
58 2
|
4月前
业务系统架构实践问题之代码应该主要放在biz层还是domain层
业务系统架构实践问题之代码应该主要放在biz层还是domain层
接口模板,文本常用的接口Controller层,常用的controller层模板,Mybatisplus的相关配置
接口模板,文本常用的接口Controller层,常用的controller层模板,Mybatisplus的相关配置
|
6月前
|
消息中间件 设计模式 监控
如何优雅地实现接口统一调用?
【2月更文挑战第6天】
393 3
|
存储 前端开发 Java
Service 层异常抛到 Controller 层处理还是直接处理?
Service 层异常抛到 Controller 层处理还是直接处理?
422 1
统一结果返回
统一结果返回,可参考
72 0