Springboot后端返回统一格式数据

本文介绍了如何在SpringBoot项目中实现前端与后端数据交互时的自动异常捕获和结果包装,通过创建异常枚举、包装类和配置全局/成功结果处理器,简化了Controller中的代码并提高了可维护性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. 为什么这么做

前端方便处理 后端返回的数据多种多样 不统一的话前端会杀人

2. 实操

我是结合了我看到的所有方案搞得一套 全自动收集异常以及自动包装结果无需在controller中调用的模式,如果需要自定义更改或删除相应文件即可

exception包下(目类结构也可以自行调整)

(1)创建异常枚举类(用来以后统一管理业务异常)
package org.example.springtest.exception;

import lombok.Getter;

@Getter
public enum AppExceptionCodeMsg {
    INVALID_CODE(10000,"验证码无效"),
    USERNAME_NOT_EXITS(10001,"用户名不存在"),
    ;
    private int code;
    private String msg;
    AppExceptionCodeMsg(int code,String msg){
        this.code=code;
        this.msg=msg;
    }
}
(2)创建异常包装类
package org.example.springtest.exception;

import lombok.Getter;
import org.example.springtest.exception.AppExceptionCodeMsg;

@Getter
public class AppException extends RuntimeException{
    private int code=500;
    private String msg="服务器异常";
    public AppException(AppExceptionCodeMsg appExceptionCodeMsg){
        super();
        this.code=appExceptionCodeMsg.getCode();
        this.msg=appExceptionCodeMsg.getMsg();
    }
    public AppException(int code,String msg){
        super();
        this.code=code;
        this.msg=msg;
    }

}

modal包下

(3)包装返回结果类
package org.example.springtest.modal;

import lombok.Getter;
import org.example.springtest.exception.AppExceptionCodeMsg;

@Getter
public class Resp<T> {
    private int code=200;
    private String msg="success";
    private final T data;
    private boolean success=true;
    private  Resp(int code,String msg,T data,boolean success){
        this.code=code;
        this.msg=msg;
        this.data=data;
        this.success=success;
    }
    public static <T> Resp success(T data){
        return new Resp(200,"success",data,true);
    }
    public static <T> Resp success(String msg,T data){
        return new Resp(200,msg,data,true);
    }
    public static <T>Resp error(int code,String msg){
        return new Resp(code,msg,null,false);
    }

    public static <T>Resp error(AppExceptionCodeMsg appExceptionCodeMsg){
        return new Resp(appExceptionCodeMsg.getCode(),appExceptionCodeMsg.getMsg(),null,false);
    }

}

config下

(4)第一部分是异常自动捕获自动包装
package org.example.springtest.config;

import org.example.springtest.exception.AppException;
import org.example.springtest.modal.Resp;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;

@ControllerAdvice
public class GlobalExceptionHandler {
    @ExceptionHandler(value = {Exception.class})
    @ResponseBody
    public <T> Resp<T> exceptionHandler(Exception e) {
        if (e instanceof AppException) {
            AppException appException=(AppException)e;
            return Resp.error(appException.getCode(),appException.getMsg());
        }
        return Resp.error(500,"服务器异常");
    }
}
(5)第二部分是成功的结果自动捕获自动包装
package org.example.springtest.config;

import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.SneakyThrows;
import org.example.springtest.modal.Resp;
import org.springframework.beans.factory.annotation.Autowired;
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.RestControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;

@RestControllerAdvice
public class ResponseAdvice implements ResponseBodyAdvice<Object> {
    @Autowired
    private ObjectMapper objectMapper;
    @Override
    public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
        return true;
    }
    @SneakyThrows
    @Override
    public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        if(o instanceof String){
            return objectMapper.writeValueAsString(Resp.success(o));
        }
        if(o instanceof Resp){
            return o;
        }
        return Resp.success(o);
    }
}

ok然后就大功告成了 上面的逻辑都比较简单也没上面多讲的 直接看效果

例子:可以看到我在controller类里是完全不调用我们的包装的

package org.example.springtest.controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class HelloController {
    @RequestMapping("/hello")
    public String Hello(){
        return "Hello Word!";
    }
}

返回结果

我们也可以抛出异常尝试一下

    @Operation(summary = "异常测试")
    @GetMapping("/exceptionTest")
    public String exceptionTest(){
        throw new AppException(AppExceptionCodeMsg.INVALID_CODE);
    }

结果

### 处理数据库日期字段的最佳实践 在 Spring Boot 后端项目中处理来自数据库的日期数据时,确保正确性和一致性非常重要。以下是最佳实践: #### 使用合适的 Java 类型表示日期时间 对于不同的 MySQL 数据库中的 `DATE`、`TIME` 和 `DATETIME/TIMESTAMP` 字段,应该分别映射到 Java 中对应的类。 - 对于仅包含日期的部分(年月日),应使用 `java.time.LocalDate`。 - 如果只需要时间部分,则推荐使用 `java.time.LocalTime`。 - 当涉及到完整的日期时间和可能存在的时区信息时,建议采用 `java.time.LocalDateTime` 或者带有时区支持的 `java.time.ZonedDateTime`[^1]。 ```java // 实体类属性声明示例 private LocalDate birthDate; // 只有日期 private LocalTime startTime; // 只有时分秒 private LocalDateTime createdDateTime; // 完整的时间戳 ``` #### 设置正确的 JDBC URL 参数 为了防止潜在的数据丢失或不一致问题,在应用程序配置文件 (`application.properties`) 中设置合理的参数是非常必要的。特别是当从较旧版本升级驱动程序时更要注意这一点。 ```properties spring.datasource.url=jdbc:mysql://localhost:3306/test?useSSL=false&serverTimezone=UTC ``` 这里指定了服务器所在的时区为 UTC 来保持统一性,并关闭 SSL 连接以简化测试环境下的配置[^3]。 #### 自动化 JSON 转换 为了让客户端能够更好地理解和解析返回的结果集里的日期对象,默认情况下 Jackson 序列化器会把它们转换成 ISO8601 格式的字符串形式发送出去。如果希望自定义这种行为的话可以通过如下方式调整全局序列化的模式: ```yaml # application.yml 文件片段 spring: jackson: date-format: yyyy-MM-dd HH:mm:ss # 设定默认格式 time-zone: GMT+8 # 显式指定时区偏移量 ``` 或者利用注解的方式针对特定实体成员变量单独设定其表现形式: ```java @JsonFormat(pattern="yyyy-MM-dd", timezone = "GMT+8") private LocalDate someField; ``` 以上方法可以帮助开发者更加高效地管理并传递日期类型的值给前端应用或其他微服务组件之间通信所必需的信息[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值