当前位置:   article > 正文

SpringBoot从入门到精通教程(二十六)- 全局header/body接口请求参数+Swagger2集成/接口规范用法_springboot header参数

springboot header参数

需求背景

在实际服务端API接口项目开发过程中,会有一些项目约定规范用法Tips,这次整理分享一下我过去使用过的,希望对你有用

问题痛点

  • 项目开发时,没有统一参数规范约定,App对接成本、代码维护成本太高
  • 过去开发人员写代码时,要写很多必须要写但是又重复的代码,比如构造函数、getter/setter方法等
  • 一个接口返回时,无论内部是返回成功、失败、异常等,都统一返回了http状态码200,导致当集成监控工具时(因为监控工具一般是检测http状态码),无法监控区分,需要二次自定义开发/写脚本等

Tips技术点

1. 接口参数规范

  • 请求服务端接口时,统一使用全局header/body接口请求参数
  • body参数统一使用DTO对象传输,使用注解@NotEmpty进行参数空校验

2. 开发代码规范

  • 使用lombok组件,让代码更简洁,实现优雅编码
  • 接口返回值需区分http状态码200

3. 集成swagger

十分简单、简洁易用的在线接口文档组件swagger

Swagger入门教程用法:SpringBoot从入门到精通教程(二十四)- Swagger集成用法

代码演示

1. 项目目录结构

2. pom.xml依赖组件

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  5. <modelVersion>4.0.0</modelVersion>
  6. <parent>
  7. <groupId>com.md</groupId>
  8. <artifactId>spring-boot2-parent</artifactId>
  9. <version>0.0.1-SNAPSHOT</version>
  10. <relativePath>../pom.xml</relativePath>
  11. </parent>
  12. <artifactId>spring-boot2-swagger-req-params</artifactId>
  13. <packaging>jar</packaging>
  14. <name>spring-boot2-swagger-req-params</name>
  15. <description>Spring Boot, MVC, Rest API for App</description>
  16. <properties>
  17. <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  18. <java.version>1.8</java.version>
  19. <swagger.version>2.9.2</swagger.version>
  20. </properties>
  21. <dependencies>
  22. <dependency>
  23. <groupId>org.springframework.boot</groupId>
  24. <artifactId>spring-boot-starter</artifactId>
  25. </dependency>
  26. <!-- 构建成可运行的Web项目 -->
  27. <dependency>
  28. <groupId>org.springframework.boot</groupId>
  29. <artifactId>spring-boot-starter-web</artifactId>
  30. </dependency>
  31. <dependency>
  32. <groupId>net.sf.json-lib</groupId>
  33. <artifactId>json-lib-ext-spring</artifactId>
  34. </dependency>
  35. <!-- swagger集成 -->
  36. <dependency>
  37. <groupId>io.springfox</groupId>
  38. <artifactId>springfox-swagger2</artifactId>
  39. <version>${swagger.version}</version>
  40. </dependency>
  41. <!-- 默认swagger-ui -->
  42. <dependency>
  43. <groupId>io.springfox</groupId>
  44. <artifactId>springfox-swagger-ui</artifactId>
  45. <version>${swagger.version}</version>
  46. </dependency>
  47. <!-- 更易用第三方swagger-ui组件 -->
  48. <dependency>
  49. <groupId>com.github.xiaoymin</groupId>
  50. <artifactId>swagger-bootstrap-ui</artifactId>
  51. <version>1.9.3</version>
  52. </dependency>
  53. <dependency>
  54. <groupId>org.projectlombok</groupId>
  55. <artifactId>lombok</artifactId>
  56. </dependency>
  57. <dependency>
  58. <groupId>com.alibaba</groupId>
  59. <artifactId>fastjson</artifactId>
  60. </dependency>
  61. </dependencies>
  62. <build>
  63. <plugins>
  64. <plugin>
  65. <groupId>org.springframework.boot</groupId>
  66. <artifactId>spring-boot-maven-plugin</artifactId>
  67. </plugin>
  68. </plugins>
  69. </build>
  70. </project>

3. 定义全局header参数

在SwaggerConfig.java文件中,可以新增全局header参数(接口定义本身不需要再重复写了),所有接口会自动带上这个参数,比如userId或者token等

我这里使用了userId,完整代码如下:

  1. package com.md.demo;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. import org.springframework.context.annotation.Bean;
  5. import org.springframework.context.annotation.Configuration;
  6. import springfox.documentation.builders.ApiInfoBuilder;
  7. import springfox.documentation.builders.ParameterBuilder;
  8. import springfox.documentation.builders.PathSelectors;
  9. import springfox.documentation.builders.RequestHandlerSelectors;
  10. import springfox.documentation.schema.ModelRef;
  11. import springfox.documentation.service.ApiInfo;
  12. import springfox.documentation.service.Parameter;
  13. import springfox.documentation.spi.DocumentationType;
  14. import springfox.documentation.spring.web.plugins.Docket;
  15. import springfox.documentation.swagger2.annotations.EnableSwagger2;
  16. @Configuration
  17. @EnableSwagger2
  18. public class SwaggerConfig {
  19. /**
  20. * 创建一个Docket对象 调用select()方法, 生成ApiSelectorBuilder对象实例,该对象负责定义外漏的API入口
  21. * 通过使用RequestHandlerSelectors和PathSelectors来提供Predicate,在此我们使用any()方法,将所有API都通过Swagger进行文档管理
  22. *
  23. * @return
  24. */
  25. @Bean
  26. public Docket createRestApi() {
  27. // 定义全局header参数
  28. ParameterBuilder useridPar = new ParameterBuilder();
  29. List<Parameter> pars = new ArrayList<>();
  30. useridPar.name("userId").defaultValue("").description("用户id").modelRef(new ModelRef("string"))
  31. .parameterType("header").required(false).build();
  32. pars.add(useridPar.build());
  33. return new Docket(DocumentationType.SWAGGER_2)
  34. .apiInfo(apiInfo()).select()
  35. //如果不想将所有的接口都通过swagger管理的话,可以将RequestHandlerSelectors.any()修改为RequestHandlerSelectors.basePackage()
  36. //.apis(RequestHandlerSelectors.any())
  37. .apis(RequestHandlerSelectors.basePackage("com.md"))
  38. .paths(PathSelectors.any())
  39. .build()
  40. .globalOperationParameters(pars);
  41. }
  42. @SuppressWarnings("deprecation")
  43. private ApiInfo apiInfo() {
  44. return new ApiInfoBuilder()
  45. // 标题
  46. .title("SpringBoot2 中使用Swagger2 构建RESTful APIs")
  47. // 简介
  48. .description("This a demo for Swagger2")
  49. // 服务条款
  50. .termsOfServiceUrl("https://blog.csdn.net/hemin1003")
  51. // 作者个人信息
  52. .contact("Minbo.He")
  53. // 版本
  54. .version("1.0").build();
  55. }
  56. }

4. 定义DTO参数对象

GetByIdDTO.java

  1. package com.md.demo.dto;
  2. import javax.validation.constraints.NotEmpty;
  3. import io.swagger.annotations.ApiModel;
  4. import io.swagger.annotations.ApiModelProperty;
  5. import lombok.Data;
  6. @Data
  7. @ApiModel("根据用户ID标识查询")
  8. public class GetByIdDTO {
  9. @ApiModelProperty(value = "用户ID标识", required = true)
  10. @NotEmpty(message = "用户ID标识不能为空")
  11. private String id;
  12. }

5. 定义一个接口基类

BaseController.java,接收参数合法性校验结果

  1. package com.md.demo.util;
  2. import java.util.List;
  3. import org.springframework.validation.BindingResult;
  4. import org.springframework.validation.ObjectError;
  5. /**
  6. * 基类
  7. *
  8. * @author Minbo
  9. *
  10. */
  11. public class BaseController {
  12. @SuppressWarnings({ "rawtypes", "unchecked" })
  13. public JsonResult getJsonResult(Object model, BindingResult result) {
  14. if (result.hasErrors()) {
  15. StringBuilder stringBuilder = new StringBuilder();
  16. List<ObjectError> allErrors = result.getAllErrors();
  17. for (ObjectError objectError : allErrors) {
  18. String defaultMessage = objectError.getDefaultMessage();
  19. // 验证失败时提示内容一起拼接返回
  20. stringBuilder.append(defaultMessage).append(";");
  21. }
  22. return new JsonResult(CodeEnums.PARA_ERR.getCode(), stringBuilder.toString(), model);
  23. }
  24. return null;
  25. }
  26. }

6. 定义一个http状态码工具类

HttpStatusCodeUtil.java

  1. package com.md.demo.util;
  2. import javax.servlet.http.HttpServletResponse;
  3. import lombok.extern.slf4j.Slf4j;
  4. /**
  5. * http响应码工具类
  6. *
  7. * @author Minbo
  8. *
  9. */
  10. @Slf4j
  11. public class HttpStatusCodeUtil {
  12. /**
  13. * 设置http响应码
  14. *
  15. * @param response
  16. * @param statusCode
  17. */
  18. public static void setCode(HttpServletResponse response, Integer statusCode) {
  19. try {
  20. response.setStatus(statusCode);
  21. } catch (Exception ex) {
  22. log.error("设置http响应码异常:" + ex.getMessage(), ex);
  23. }
  24. }
  25. }

7. 接口访问层(@RequestHeader和@RequestBody

InitController.java

  1. package com.md.demo.rest;
  2. import org.slf4j.Logger;
  3. import org.slf4j.LoggerFactory;
  4. import org.springframework.web.bind.annotation.RequestHeader;
  5. import org.springframework.web.bind.annotation.RequestMapping;
  6. import org.springframework.web.bind.annotation.RestController;
  7. import com.md.demo.util.JsonResult;
  8. import io.swagger.annotations.ApiImplicitParam;
  9. import io.swagger.annotations.ApiOperation;
  10. /**
  11. * @author Minbo
  12. */
  13. @RestController
  14. public class InitController {
  15. protected static Logger logger = LoggerFactory.getLogger(InitController.class);
  16. /**
  17. * http://localhost:9090/hello
  18. *
  19. * @return
  20. */
  21. @ApiOperation(value = "/hello 欢迎入口", httpMethod = "GET")
  22. @RequestMapping(value = "/hello")
  23. public String hello() {
  24. logger.info("hello");
  25. return "Hello greetings from spring-boot2-swagger-req-params";
  26. }
  27. @ApiOperation(value = "/getUserName 根据用户id获得用户的姓名", notes = "id不能为空", httpMethod = "GET")
  28. @ApiImplicitParam(dataType = "string", name = "userId", value = "用户id", required = true)
  29. @RequestMapping(value = "/getUserName")
  30. @SuppressWarnings({ "rawtypes" })
  31. public JsonResult getUserName(@RequestHeader String userId) {
  32. String result = "hello " + userId + ",name=张三";
  33. return JsonResult.ok(result);
  34. }
  35. /**
  36. * Swagger注解用法:
  37. *
  38. * @Api:修饰整个类,描述Controller的作用
  39. * @ApiOperation:描述一个类的一个方法,或者说一个接口
  40. * @ApiParam:单个参数描述
  41. * @ApiModel:用对象来接收参数
  42. * @ApiProperty:用对象接收参数时,描述对象的一个字段
  43. * @ApiResponse:HTTP响应其中1个描述
  44. * @ApiResponses:HTTP响应整体描述
  45. * @ApiIgnore:使用该注解忽略这个API
  46. * @ApiError :发生错误返回的信息
  47. * @ApiImplicitParam:一个请求参数
  48. * @ApiImplicitParams:多个请求参数
  49. */
  50. }

DemoController.java

  1. package com.md.demo.rest;
  2. import javax.servlet.http.HttpServletRequest;
  3. import javax.servlet.http.HttpServletResponse;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.validation.BindingResult;
  6. import org.springframework.validation.annotation.Validated;
  7. import org.springframework.web.bind.annotation.PostMapping;
  8. import org.springframework.web.bind.annotation.RequestBody;
  9. import org.springframework.web.bind.annotation.RequestMapping;
  10. import org.springframework.web.bind.annotation.RestController;
  11. import com.md.demo.dto.GetByIdDTO;
  12. import com.md.demo.service.IUserService;
  13. import com.md.demo.util.BaseController;
  14. import com.md.demo.util.HttpStatusCodeUtil;
  15. import com.md.demo.util.JsonResult;
  16. import com.md.demo.vo.UserVO;
  17. import io.swagger.annotations.Api;
  18. import io.swagger.annotations.ApiOperation;
  19. import lombok.extern.slf4j.Slf4j;
  20. /**
  21. * @author Minbo
  22. */
  23. @RestController
  24. @RequestMapping("/demo")
  25. @Api(tags = { "接口-演示" })
  26. @Slf4j
  27. public class DemoController extends BaseController {
  28. @Autowired
  29. private IUserService userService;
  30. @ApiOperation(value = "根据id获得用户信息", notes = "id不能为空", httpMethod = "POST")
  31. @PostMapping(value = "/getUserById")
  32. @SuppressWarnings({ "rawtypes", "unchecked" })
  33. public JsonResult<UserVO> getUserById(@Validated @RequestBody GetByIdDTO dto, BindingResult result,
  34. HttpServletRequest request, HttpServletResponse response) {
  35. // 验证参数合法性(@NotEmpty注解)
  36. JsonResult validResult = super.getJsonResult(dto, result);
  37. if (validResult != null) {
  38. // 设置http响应码,利于监控工具,不要统一使用200
  39. HttpStatusCodeUtil.setCode(response, 4403);
  40. return validResult;
  41. }
  42. // 获取用户ID
  43. String userId = request.getHeader("userId");
  44. log.info("获取用户ID,userId={}", userId);
  45. // 逻辑处理
  46. UserVO obj = this.userService.getUserById(dto);
  47. return JsonResult.ok(obj);
  48. }
  49. }

说明:

  • 使用了注解@Validated,验证@NotEmpty
  • 使用了BindingResult对象,接收参数合法性检查结果
  • 如果参数校验不通过,则自定义设置http响应码
  • 使用了@Slf4j注解,可自动集成log日志输出

接口测试

1. 接口自动增加了全局header参数

2. 接口body参数用法,以及自动附加了接口数据模型说明

返回数据模型说明:UserVO.java类

3. 接口成功时:响应返回内容

各个字段及内容,会自动映射对应上,客户端接入时简单、高效

4. 接口异常时:响应返回内容

完整源码下载

我的Github源码地址:

https://github.com/hemin1003/spring-boot-study/tree/master/spring-boot2-study/spring-boot2-parent/spring-boot2-swagger-req-params

下一章教程

SpringBoot从入门到精通教程(二十七)- @Valid注解用法详解+全局处理器Exception优雅处理参数验证用法

该系列教程

SpringBoot从入门到精通教程

Swagger入门教程用法:SpringBoot从入门到精通教程(二十四)- Swagger集成用法

我的专栏

 

 

至此,全部介绍就结束了

 

 

-------------------------------

-------------------------------

 

我的CSDN主页

关于我(个人域名)

我的开源项目集Github

 

期望和大家一起学习,一起成长,共勉,O(∩_∩)O谢谢

欢迎交流问题,可加个人QQ 469580884,

或者,加我的群号 751925591,一起探讨交流问题

不讲虚的,只做实干家

Talk is cheap,show me the code

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Monodyee/article/detail/262849
推荐阅读
相关标签
  

闽ICP备14008679号