项目搭建模板

1. Spring Boot 项目搭建基础模板

建立一个全新的项目,或者把旧的庞大的项目,进行拆分成多个项目。在建立新的项目中,经常需要做一些重复的工作,比如说拷贝一下常用的工具类,通用代码等等。所以就可以做一个基础的项目方便使用,在经历新项目的时候,直接在基础项目上进行简单配置就可以开发业务代码了
基础项目该包含哪些东西。

  • Swagger在线接口文档。
  • CodeGenerator 代码生成器。
  • 统一返回。
  • 通用的分页对象。
  • 常用工具类。
  • 全局异常拦截。
  • 错误枚举。
  • 自定义异常。
  • 多环境配置文件。
  • Maven多环境配置。
  • 日志配置。
  • JenkinsFile。

    1.1. 在线接口文档 Swagger

    1.1.1. 简介

    Swagger可以在线生成接口文档,并且可以在页面上进行测试。
    00-项目搭建模板 - 图1
    可以非常清楚的显示,请求数据已经响应数据。当然这一切都需要在代码中进行配置。
    00-项目搭建模板 - 图2
    注意的点:接口文档只能在测试/开发环境开启,其他环境请关闭。

    1.1.2. 常用的Swagger注解

    @Api用于Controller
    @ApiOperation用于Controller内的方法
    @ApiResponses用于标识接口返回数据的类型
    @ApiModel用于标识类的名称
    @ApiModelProperty用于标识属性的名称
    对应的swagger的配置可以查看基础项目内的SwaggerConfiguration.java ``` @RestController @Api(tags = “用户”) @AllArgsConstructor @RequestMapping(“/user”) public class UserController { private IUserService userService;

    @ApiOperation(“获取用户列表”) @GetMapping(“/listUser”) @ApiResponses(

    1. @ApiResponse(code = 200, message = "操作成功", response = UserVo.class)

    ) public ResultVo listUser(@Validated ListUserForm listUserForm){

    1. return ResultVoUtil.success(userService.listUser(listUserForm));

    } }

  1. ```
  2. @Data
  3. @ApiModel("获取用户列表需要的表单数据")
  4. @EqualsAndHashCode(callSuper = false)
  5. public class ListUserForm extends PageForm<ListUserForm> {
  6. @ApiModelProperty("用户状态")
  7. @NotEmpty(message = "用户状态不能为空")
  8. @Range(min = -1 , max = 1 , message = "用户状态有误")
  9. private String status;
  10. }

1.2. CodeGenerator 代码生成器

mybatis_plus代码生成器可以一键生成entity,service,serviceImpl,mapper,mapper.xml。省去了建立一大堆实体类的麻烦。
由于配置太长这里就不贴出来了,对应的CodeGenerator的配置可以查看基础项目内的CodeGenerator.java

1.3. 常用的封装

1.3.1. 统一返回 ResultVo

将所有的接口的响应数据的格式进行统一

  1. @Data
  2. @ApiModel("固定返回格式")
  3. public class ResultVo {
  4. @ApiModelProperty("错误码")
  5. private Integer code;
  6. @ApiModelProperty("提示信息")
  7. private String message;
  8. @ApiModelProperty("响应数据")
  9. private Object data;
  10. }

1.3.2. 创建抽象基础请求表单类 BaseForm

很多情况都是将接受到的参数,转变成对应的实体类然后保存或者更新。所以对于这类的form可以继承baseform并实现buildEntity()方法,这样可以更加符合面向对象,service不需要关心请求参数form如何转变成entity,只需要在使用的时候调用buildEntity()即可,尤其是在form转成entity相对复杂的时候,这样做可以减少service内的代码。让代码逻辑看起来更加清晰。
创建抽象基础请求表单类BaseForm<T>

  1. public abstract class BaseForm<T> {
  2. public abstract T buildEntity();
  3. }

示例,通过使用BaseForm抽象父类进行重构 AddUserForm 继承 BaseForm并重写buildEntity()方法,原逻辑如下

  1. @Override
  2. public boolean addUser(AddUserForm userForm) {
  3. User user = new User();
  4. user.setNickname(userForm.getNickname());
  5. user.setBirthday(userForm.getBirthday());
  6. user.setUsername(userForm.getUsername());
  7. user.setPassword(userForm.getPassword());
  8. return save(user);
  9. }
  1. @Data
  2. @EqualsAndHashCode(callSuper = false)
  3. public class AddUserForm extends BaseForm<User> {
  4. private String nickname;
  5. private Date birthday;
  6. private String username;
  7. private String password;
  8. @Override
  9. public User buildEntity() {
  10. User user = new User();
  11. BeanUtils.copyProperties(this,user);
  12. return user;
  13. }
  14. }
  15. @Override
  16. public boolean addUser(AddUserForm userForm) {
  17. return save(userForm.buildEntity());
  18. }

1.3.3. 通用的分页对象

查询的时候,绝大多数都需要用到分页,所以说封装分页对象就很有必要。可以注意下PageForm.calcCurrent()PageVo.setCurrentAndSize()PageVo.setTotal()这个几个方法

  • 分页请求实体PageForm ``` @Data @ApiModel(value = “分页数据”, description = “分页需要的表单数据”) public class PageForm>{

    @ApiModelProperty(value = “页码 从第一页开始 1”) @Min(value = 1, message = “页码输入有误”) private Integer current;

    @ApiModelProperty(value = “每页显示的数量 范围在1~100”) @Range(min = 1, max = 100, message = “每页显示的数量输入有误”) private Integer size;

    @ApiModelProperty(hidden = true) public T calcCurrent(){

    1. current = (current - 1 ) * size;
    2. return (T) this;

    } }

  1. - 响应分页实体`PageVo`

@Data public class PageVo {

  1. @ApiModelProperty(value = "分页数据")
  2. private List<T> records;
  3. @ApiModelProperty(value = "总条数")
  4. private Integer total;
  5. @ApiModelProperty(value = "总页数")
  6. private Integer pages;
  7. @ApiModelProperty(value = "当前页")
  8. private Integer current;
  9. @ApiModelProperty(value = "查询数量")
  10. private Integer size;
  11. @ApiModelProperty(hidden = true)
  12. public PageVo<T> setCurrentAndSize(PageForm<?> pageForm){
  13. BeanUtils.copyProperties(pageForm,this);
  14. return this;
  15. }
  16. @ApiModelProperty(hidden = true)
  17. public void setTotal(Integer total) {
  18. this.total = total;
  19. this.setPages(this.total % this.size > 0 ? this.total / this.size + 1 : this.total / this.size);
  20. }

}

  1. **案例**
  2. - ListUserForm

@Data @ApiModel(“获取用户列表需要的表单数据”) @EqualsAndHashCode(callSuper = false) public class ListUserForm extends PageForm { @ApiModelProperty(“用户状态”) @NotEmpty(message = “用户状态不能为空”) @Range(min = -1 , max = 1 , message = “用户状态有误”) private String status; }

  1. - UserServiceImpl

@Override public PageVo listUser(ListUserForm listUserForm) { PageVo pageVo = new PageVo().setCurrentAndSize(listUserForm); pageVo.setTotal(countUser(listUserForm.getStatus())); pageVo.setRecords(userMapper.listUser(listUserForm.calcCurrent())); return pageVo; }

private Integer countUser(String status){ return count(new QueryWrapper().eq(“status”,status)); }

  1. - UserController

@ApiOperation(“获取用户列表”) @GetMapping(“/listUser”) @ApiResponses( @ApiResponse(code = 200, message = “操作成功”, response = UserVo.class) ) public ResultVo listUser(@Validated ListUserForm listUserForm){ return ResultVoUtil.success(userService.listUser(listUserForm)); }

``` 00-项目搭建模板 - 图3
案例注意点说明:

  • PageVo在实例化的时候需要设置当前页和每页显示的数量,可以调用setCurrentAndSize()完成。
  • 进行分页查询的时候,需要计算偏移量。listUserForm.calcCurrent()

为什么要计算偏移量呢?

  • 假如查询第1页每页显示10条记录,前端传递过来的参数是current=1&&size=10,这个时候limit 1,10没有问题。
  • 假如查询第2页每页显示10条记录,前端传递过来的参数是current=2&&size=10,这个时候limit 2,10就有问题,实际应该是limit 10,10calcCurrent()方法的作用就是如此。

    1.4. 常用工具类

    常用工具类可以根据平时的项目积累与开发习惯引入。

    1.5. 异常处理