从零搭建Spring Boot脚手架(2):增加通用的功能(转载)

  1. 前言
    上一篇说了我要一步步地搭建Spring Boot脚手架,首先会集成Spring MVC并进行定制化以满足日常开发的需要,我们先做一些刚性的需求定制,后续再补充细节。如果你看了本文有什么问题可以留言讨论。多多持续关注,共同学习,共同进步。

Gitee:https://gitee.com/felord/kono

GitHub: https://github.com/NotFound403/kono

  1. 统一返回体
    在开发中统一返回数据非常重要。方便前端统一处理。通常设计为以下结构:

{
“code”: 200,
“data”: {
“name”: “felord.cn”,
“age”: 18
},
“msg”: “”,
“identifier”: “”
}
code 业务状态码,设计时应该区别于 http 状态码。

data 数据载体,用以装载返回给前端展现的数据。

msg 提示信息,用于前端调用后返回的提示信息,例如 “新增成功”、“删除失败”。

identifier 预留的标识位,作为一些业务的处理标识。

根据上面的一些定义,声明了一个统一返回体对象RestBody并声明了一些静态方法来方便定义。

package cn.felord.kono.advice;

import lombok.Data;

import java.io.Serializable;

/**

  • @author felord.cn

  • @since 22:32 2019-04-02
    */
    @Data
    public class RestBody implements Rest, Serializable {

    private static final long serialVersionUID = -7616216747521482608L;
    private int code = 200;
    private T data;
    private String msg = “”;
    private String identifier = “”;

    public static Rest<?> ok() {
    return new RestBody<>();
    }

    public static Rest<?> ok(String msg) {
    Rest<?> restBody = new RestBody<>();
    restBody.setMsg(msg);
    return restBody;
    }

    public static Rest okData(T data) {
    Rest restBody = new RestBody<>();
    restBody.setData(data);
    return restBody;
    }

    public static Rest okData(T data, String msg) {
    Rest restBody = new RestBody<>();
    restBody.setData(data);
    restBody.setMsg(msg);
    return restBody;
    }

    public static Rest build(int code, T data, String msg, String identifier) {
    Rest restBody = new RestBody<>();
    restBody.setCode(code);
    restBody.setData(data);
    restBody.setMsg(msg);
    restBody.setIdentifier(identifier);
    return restBody;
    }

    public static Rest<?> failure(String msg, String identifier) {
    Rest<?> restBody = new RestBody<>();
    restBody.setMsg(msg);
    restBody.setIdentifier(identifier);
    return restBody;
    }

    public static Rest<?> failure(int httpStatus, String msg ) {
    Rest<?> restBody = new RestBody< >();
    restBody.setCode(httpStatus);
    restBody.setMsg(msg);
    restBody.setIdentifier("-9999");
    return restBody;
    }

    public static Rest failureData(T data, String msg, String identifier) {
    Rest restBody = new RestBody<>();
    restBody.setIdentifier(identifier);
    restBody.setData(data);
    restBody.setMsg(msg);
    return restBody;
    }

    @Override
    public String toString() {
    return “{” +
    “code:” + code +
    “, data:” + data +
    “, msg:” + msg +
    “, identifier:” + identifier +
    ‘}’;
    }
    }
    但是每次都要显式声明返回体也不是很优雅的办法,所以我们希望无感知的来实现这个功能。Spring Framework正好提供此功能,我们借助于@RestControllerAdvice和ResponseBodyAdvice来对项目的每一个@RestController标记的控制类的响应体进行后置切面通知处理。

/**

  • 统一返回体包装器

  • @author felord.cn

  • @since 14:58
    **/
    @RestControllerAdvice
    public class RestBodyAdvice implements ResponseBodyAdvice {

    @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 == null) {
    return RestBody.ok();
    }
    // 如果 RestBody 的 父类 是 返回值的父类型 直接返回
    // 方便我们可以在接口方法中直接返回RestBody
    if (Rest.class.isAssignableFrom(o.getClass())) {
    return o;
    }
    // 进行统一的返回体封装
    return RestBody.okData(o);
    }
    }
    当我们接口返回一个实体类时会自动封装到统一返回体RestBody中。

既然有ResponseBodyAdvice,就有一个RequestBodyAdvice,它似乎是来进行前置处理的,以后可能有一些用途。

  1. 统一异常处理
    统一异常也是@RestControllerAdvice能实现的,可参考之前的Hibernate Validator 校验参数全攻略。这里初步集成了校验异常的处理,后续会添加其他异常。

/**

  • 统一异常处理

  • @author felord.cn

  • @since 13 :31 2019-04-11
    */
    @Slf4j
    @RestControllerAdvice
    public class ApiExceptionHandleAdvice {

    @ExceptionHandler(BindException.class)
    public Rest<?> handle(HttpServletRequest request, BindException e) {
    logger(request, e);
    List allErrors = e.getAllErrors();
    ObjectError objectError = allErrors.get(0);
    return RestBody.failure(700, objectError.getDefaultMessage());
    }

    @ExceptionHandler(MethodArgumentNotValidException.class)
    public Rest<?> handle(HttpServletRequest request, MethodArgumentNotValidException e) {
    logger(request, e);
    List allErrors = e.getBindingResult().getAllErrors();
    ObjectError objectError = allErrors.get(0);
    return RestBody.failure(700, objectError.getDefaultMessage());
    }

    @ExceptionHandler(ConstraintViolationException.class)
    public Rest<?> handle(HttpServletRequest request, ConstraintViolationException e) {
    logger(request, e);
    Optional<ConstraintViolation<?>> first = e.getConstraintViolations().stream().findFirst();
    String message = first.isPresent() ? first.get().getMessage() : “”;
    return RestBody.failure(700, message);
    }

    @ExceptionHandler(Exception.class)
    public Rest<?> handle(HttpServletRequest request, Exception e) {
    logger(request, e);
    return RestBody.failure(700, e.getMessage());
    }

    private void logger(HttpServletRequest request, Exception e) {
    String contentType = request.getHeader(“Content-Type”);
    log.error(“统一异常处理 uri: {} content-type: {} exception: {}”, request.getRequestURI(), contentType, e.toString());
    }
    }

  1. 简化类型转换
    简化Java Bean之间转换也是一个必要的功能。这里选择mapStruct,类型安全而且容易使用,比那些BeanUtil要好用的多。但是从我使用的经验上来看,不要使用mapStruct提供的复杂功能只做简单映射。详细可参考文章Spring Boot 2 实战:集成 MapStruct 类型转换。

集成进来非常简单,由于它只在编译期生效所以引用时的scope应该设置为compile,我们在kono-dependencies中加入其依赖管理:

org.mapstruct mapstruct ${mapstruct.version} compile org.mapstruct mapstruct-processor ${mapstruct.version} compile 在kono-app中直接引用上面两个依赖,但是这样还不行,和lombok一起使用编译容易出现SPI错误。我们还需要集成相关的 Maven 插件到kono-app编译的生命周期中去。参考如下: org.apache.maven.plugins maven-compiler-plugin 3.8.1 1.8 1.8 true org.projectlombok lombok ${lombok.version} org.mapstruct mapstruct-processor ${mapstruct.version} 然后我们就很容易将一个Java Bean转化为另一个Java Bean。下面这段代码将UserInfo转换为UserInfoVO而且自动为UserInfoVO.addTime赋值为当前时间,同时这个工具也自动注入了Spring IoC,而这一切都发生在编译期。

编译前:

/**

  • @author felord.cn

  • @since 16:09
    **/
    @Mapper(componentModel = “spring”, imports = {LocalDateTime.class})
    public interface BeanMapping {

    @Mapping(target = “addTime”, expression = “java(LocalDateTime.now())”)
    UserInfoVO toUserInfoVo(UserInfo userInfo);

}
编译后:

package cn.felord.kono.beanmapping;

import cn.felord.kono.entity.UserInfo;
import cn.felord.kono.entity.UserInfoVO;
import java.time.LocalDateTime;
import javax.annotation.Generated;
import org.springframework.stereotype.Component;

@Generated(
value = “org.mapstruct.ap.MappingProcessor”,
date = “2020-07-30T23:11:24+0800”,
comments = “version: 1.3.0.Final, compiler: javac, environment: Java 1.8.0_252 (AdoptOpenJDK)”
)
@Component
public class BeanMappingImpl implements BeanMapping {

@Override
public UserInfoVO toUserInfoVo(UserInfo userInfo) {
    if ( userInfo == null ) {
        return null;
    }

    UserInfoVO userInfoVO = new UserInfoVO();

    userInfoVO.setName( userInfo.getName() );
    userInfoVO.setAge( userInfo.getAge() );

    userInfoVO.setAddTime( LocalDateTime.now() );

    return userInfoVO;
}

}
其实mapStruct也就是帮我们写了Getter和Setter,但是不要使用其比较复杂的转换,会增加学习成本和可维护的难度。

  1. 单元测试
    将以上功能集成进去后分别做一个单元测试,全部通过。

    @Autowired
    MockMvc mockMvc;
    @Autowired
    BeanMapping beanMapping;

    /**

    • 测试全局异常处理.

    • @throws Exception the exception

    • @see UserController#getUserInfo()
      */
      @Test
      void testGlobalExceptionHandler() throws Exception {

      String rtnJsonStr = “{\n” +
      " “code”: 700,\n" +
      " “data”: null,\n" +
      " “msg”: “test global exception handler”,\n" +
      " “identifier”: “-9999”\n" +
      “}”;

      mockMvc.perform(MockMvcRequestBuilders.get("/user/get"))
      .andExpect(MockMvcResultMatchers.content()
      .json(rtnJsonStr))
      .andDo(MockMvcResultHandlers.print());
      }

    /**

    • 测试统一返回体.
    • @throws Exception the exception
    • @see UserController#getUserVO()
      */
      @Test
      void testUnifiedReturnStruct() throws Exception {
      // “{“code”:200,“data”:{“name”:“felord.cn”,“age”:18,“addTime”:“2020-07-30T13:08:53.201”},“msg”:”",“identifier”:""}";
      mockMvc.perform(MockMvcRequestBuilders.get("/user/vo"))
      .andExpect(MockMvcResultMatchers.jsonPath(“code”, Is.is(200)))
      .andExpect(MockMvcResultMatchers.jsonPath(“data.name”, Is.is(“felord.cn”)))
      .andExpect(MockMvcResultMatchers.jsonPath(“data.age”, Is.is(18)))
      .andExpect(MockMvcResultMatchers.jsonPath(“data.addTime”, Is.is(notNullValue())))
      .andDo(MockMvcResultHandlers.print());
      }

    /**

    • 测试 mapStruct类型转换.

    • @see BeanMapping
      */
      @Test
      void testMapStruct() {
      UserInfo userInfo = new UserInfo();
      userInfo.setName(“felord.cn”);
      userInfo.setAge(18);
      UserInfoVO userInfoVO = beanMapping.toUserInfoVo(userInfo);

      Assertions.assertEquals(userInfoVO.getName(), userInfo.getName());
      Assertions.assertNotNull(userInfoVO.getAddTime());
      }

  2. 总结
    自制脚手架初步具有了统一返回体、统一异常处理、快速类型转换,其实参数校验也已经支持了。后续就该整合数据库了,常用的数据库访问技术主要为Mybatis、Spring Data JPA、JOOQ等,不知道你更喜欢哪一款?欢迎留言讨论。

往期推荐:

从零搭建Spring Boot脚手架(1):开篇以及技术选型

2020-07-30

Spring Data R2DBC响应式操作MySQL

2020-07-28

Spring Security 实战干货:从零手写一个验证码登录

2020-07-22

原文链接:https://blog.csdn.net/qq_35067322/article/details/107724570?utm_medium=distribute.pc_feed.none-task-blog-personrec_tag-2.nonecase&depth_1-utm_source=distribute.pc_feed.none-task-blog-personrec_tag-2.nonecase&request_id=5f31e26c8c9fb674c6723b49

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值