springBoot返回xml格式结果

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了springBoot返回xml格式结果相关的知识,希望对你有一定的参考价值。

参考技术A 虽然现在json已经相当普及了,但是开发过程中总会存在些向后兼容的问题,所以有时候接口需要xml返回结果又不可避免,这里说明一下如何让springBoot应用返回xml格式结果。

超级简单只需要两步:

第一步,在build.gradle配置文件中增加以下配置(如果使用maven请在对应的pom.xml配置对应信息)

compile group: 'com.fasterxml.jackson.dataformat', name: 'jackson-dataformat-xml', version: '2.8.8'

第二步,在我们的restControler类对应方法中增加@RequestMapping(value="/getProductionDictX",produces = "application/xml","application/json")

publicMessage getProductionDictX()

其中"application/xml","application/json"表示可以返回xml格式或者json格式

此时我们在浏览器中输入http://localhost:8082/getProductionDict.xml,即可以返回xml格式数据

输入http://localhost:8082/getProductionDict.json返回json格式数据。

此处有一个待解决问题,就是如果返回xml格式数据需要以下类似头信息

<?xml version="1.0" encoding="UTF-8"?>

<Message xmlns="http://nm.cmc.com/CSCS/BW/BWMethod/BW00001/">

是无法得到的。

此处如果引入compilegroup:'org.codehaus.woodstox',name:'woodstox-core-asl',version:'4.4.1'包

并在对应值对象所在目录中增加package-info.java文件,在文件中增加如下信息

@XmlSchema(

namespace ="http://nm.cmc.com/CSCS/BW/BWMethod/BW00001/",

elementFormDefault = XmlNsForm.QUALIFIED)

packagecom.esop.productInfo.design.dto;

importjavax.xml.bind.annotation.XmlNsForm;

importjavax.xml.bind.annotation.XmlSchema;

在对应的值对象类中要增加@XmlRootElement注解,如下

importjavax.xml.bind.annotation.XmlRootElement;

@XmlRootElement

public classEp_prd_production_dict

此时再访问服务地址会增加xmlns信息。结果如下

但是仍然无法解决<?xml version="1.0" encoding="UTF-8"?>这个头信息。

使用woodstox-core-asl还有一个问题,就是返回结果如果是List对象时转换成xml格式文件时会报错,所以如果返回结果是多条记录时此处不能用List对象直接返回。

参考文档:

https://docs.spring.io/spring-boot/docs/current/reference/html/howto-spring-mvc.html

http://www.jianshu.com/p/4477efb1accf

SpringBoot返回统一的JSON标准格式

  1. 自定义状态码枚举类
  2. 封装返回结果
  3. 全局异常捕获处理,使用@RestControllerAdvice注解
  4. 拦截Controller方法的返回值,统一处理返回值/响应体
  5. 创建Controller,准备测试
  6. 请求接口,查看响应结果

近年来,随着移动互联网的发展,各种类型的客户端层出不穷。如果不统一数据接口,则会造成冗余编码,增加成本。RESTful风格的API正适合通过一套统一的接口为PC、手机APP等设备提供数据服务。

为了保障前后端数据交换的顺畅,建议规范数据的返回,并采用固定的数据格式封装。如:
返回成功信息的JSON格式如下

{
  "code": 200,
  "msg": "操作成功",
  "data": "hello jenkins"
}

返回异常信息的JSON格式如下

{
  "code": 500,
  "msg": "系统异常,请稍后重试:/ by zero"
}

实现步骤如下

1. 自定义状态码枚举类。

@AllArgsConstructor
@Getter
public enum StatusCodeEnum {
    SC200(200, "操作成功"),
    SC999(999, "操作失败"),
    SC401(401, "匿名用户访问权限资源时的异常"),
    SC403(403, "无访问权限,请联系管理员授予权限"),
    SC404(404, "请求的资源不存在"),
    SC500(500, "系统异常,请稍后重试"),
    // ...略
    private final Integer code;
    private final String msg;
}

2. 封装返回结果

@Data
public class ApiResult<T> implements Serializable {
    private Integer code;
    private String msg;
    private T data;

    public static <T> ApiResult<T> success(T data) {
        return ApiResult.success(StatusCodeEnum.SC200.getMsg(), data);
    }

    public static <T> ApiResult<T> success(String msg, T data) {
        ApiResult<T> apiResult = new ApiResult<>();
        apiResult.setCode(StatusCodeEnum.SC200.getCode());
        apiResult.setMsg(msg);
        apiResult.setData(data);
        return apiResult;
    }

    public static <T> ApiResult<T> fail(Integer code, String msg) {
        ApiResult<T> apiResult = new ApiResult<>();
        apiResult.setCode(code);
        apiResult.setMsg(msg);
        return apiResult;
    }
}

3. 全局异常捕获处理,使用@RestControllerAdvice注解。

@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {
    /**
     * 捕获其他异常
     */
    @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    @ExceptionHandler(Exception.class)
    public ApiResult<String> handle(Exception e) {
        log.error("全局异常信息:{}", e.getMessage());
        return ApiResult.fail(StatusCodeEnum.SC500.getCode(), StatusCodeEnum.SC500.getMsg() + ":" + e.getMessage());
    }
}
注解功能
@RestControllerAdviceRestController的增强类,可用于实现全局异常处理器
@ExceptionHandler统一处理某一类异常,从而减少代码重复率和复杂度,比如要获取自定义异常可以@ExceptionHandler(BusinessException.class)
@ResponseStatus指定客户端收到的http状态码

注:请求进来 会按照 filter -> interceptor -> controllerAdvice -> aspect -> controller的顺序调用,
404异常(NoHandlerFoundException)是无法通过这种方式捕获的,因为在Filter层发生的异常都会到Spring默认的异常处理。如果你在配置文件配置了server.error.path的话,就会使用你配置的异常处理地址,如果没有就会使用你配置的error.path路径地址,如果还是没有,默认使用/error来作为发生异常的处理地址。如果想要替换默认的非Controller异常处理直接实现Spring提供的ErrorController接口就行了。

第四步是重点

4. 拦截Controller方法的返回值,统一处理返回值/响应体。

因为我们后面每写一个接口都需要调用ApiResult.success()这行代码对结果进行包装,重复劳动,浪费体力,我们只需要实现SpringBoot提供的ResponseBodyAdvice接口即可。

@RestControllerAdvice
public class ApiResultWrapper implements ResponseBodyAdvice<Object> {
    /**
     * 是否支持advice功能
     */
    @Override
    public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
        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 JSON.toJSONString(ApiResult.success(o));
        }
        // 这个判断的作用:防止全局异常处理后返回的结果(类型为ApiResult)再次被包装
        if (o instanceof ApiResult) {
            return o;
        }
        return ApiResult.success(o);
    }
}

5. 创建Controller,定义两个方法,让第二个方法抛异常

@RestController
public class TestController {
    @GetMapping("/test1")
    public String test1() {
        return "当前时间:" + LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
    }

    @GetMapping("/test2")
    public Integer test2() {
        System.out.println(1 / 0);
        return Integer.MAX_VALUE;
    }
}

6. 测试

分别请求http://localhost:8080/test1、http://localhost:8080/test2,结果如下

在全局异常处理类中写了一行代码

log.error("全局异常信息:{}", e.getMessage());

所以调用test2方法时控制台打印异常信息如下

以上是关于springBoot返回xml格式结果的主要内容,如果未能解决你的问题,请参考以下文章

Spring Boot 返回 XML 数据,一分钟搞定!

上手spring boot项目之springboot如何返回json数据

Springboot学习问题记录

Spring Boot Mvc 统一返回结果

springBoot项目设置统一响应返回

Spring Boot 2.x基础教程:如何扩展XML格式的请求和响应