为什么要用Swagger ?

作为一名程序员,我们最讨厌两件事:1. 别人不写注释。2. 自己写注释。

而作为一名接口开发者,我们同样讨厌两件事:1. 别人不写接口文档,文档不及时更新。2. 需要自己写接口文档,还需要及时更新。

相信无论是前端还是后端开发,都或多或少地被接口文档折磨过。前端经常抱怨后端给的接口文档与实际情况不一致。后端又觉得编写及维护接口文档会耗费不少精力,经常来不及更新。

而随着Springboot、Springcloud等微服务的流行,每个项目都有成百上千个接口调用,这时候再要求人工编写接口文档并且保证文档的实时更新几乎是一件不可能完成的事,所以这时候我们迫切需要一个工具,一个能帮我们自动化生成接口文档以及自动更新文档的工具。它就是Swagger。

Swagger 提供了一个全新的维护 API 文档的方式,有4大优点:

  1. 自动生成文档:只需要少量的注解,Swagger 就可以根据代码自动生成 API 文档,很好的保证了文档的时效性。

  2. 跨语言性,支持 40 多种语言。

  3. Swagger UI 呈现出来的是一份可交互式的 API 文档,我们可以直接在文档页面尝试 API 的调用,省去了准备复杂的调用参数的过程。

  4. 还可以将文档规范导入相关的工具(例如 SoapUI), 这些工具将会为我们自动地创建自动化测试。

现在我们知道了Swagger的作用,接下来将其集成到我们项目中。

 

Swagger集成

集成Swagger很简单,只需要简单三步。

第一步:引入依赖包

<!--swagger-->
<dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger2</artifactId><version>2.9.2</version>
</dependency><!--swagger-ui-->
<dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger-ui</artifactId><version>2.9.2</version>
</dependency>

第二步:修改配置文件

  1. application.properties 加入配置

# 用于控制是否开启Swagger,生产环境记得关闭Swagger,将值设置为 false
springfox.swagger2.enabled = true

2.增加一个swagger配置类

@Configuration
@EnableSwagger2
@ConditionalOnClass(Docket.class)
public class SwaggerConfig {private static final String VERSION = "1.0";@Value("${springfox.swagger2.enabled}")private Boolean swaggerEnabled;@Beanpublic Docket createRestApi(){return new Docket(DocumentationType.SWAGGER_2).enable(swaggerEnabled).groupName("SwaggerDemo").apiInfo(apiInfo()).select().apis(RequestHandlerSelectors.withClassAnnotation(Api.class)).paths(PathSelectors.any()).build();}/*** 添加摘要信息*/private ApiInfo apiInfo() {return new ApiInfoBuilder().title("接口文档").contact(new Contact("JAVA日知录","http://javadaily.cn","jianzh5@163.com")).description("Swagger接口文档").version(VERSION).build();}}

这里通过.apis(RequestHandlerSelectors.withClassAnnotation(Api.class))标明给加上@Api注解的类自动生成接口文档。

第三步,配置API接口

@RestController
@Api(tags = "参数校验")
@Slf4j
@Validated
public class ValidController {@PostMapping("/valid/test1")@ApiOperation("RequestBody校验")public String test1(@Validated @RequestBody ValidVO validVO){log.info("validEntity is {}", validVO);return "test1 valid success";}@ApiOperation("Form校验")@PostMapping(value = "/valid/test2")public String test2(@Validated ValidVO validVO){log.info("validEntity is {}", validVO);return "test2 valid success";}@ApiOperation("单参数校验")@PostMapping(value = "/valid/test3")public String test3(@Email String email){log.info("email is {}", email);return "email valid success";}
}

通过@Api注解标注需要生成接口文档,通过@ApiOperation注解标注接口名。

同时我们给ValidVO也加上对应的注解

@Data
@ApiModel(value = "参数校验类")
public class ValidVO {@ApiModelProperty("ID")private String id;@ApiModelProperty(value = "应用ID",example = "cloud")private String appId;@NotEmpty(message = "级别不能为空")@ApiModelProperty(value = "级别")private String level;@ApiModelProperty(value = "年龄")private int age;...}

通过@ApiModel标注这是一个参数实体,通过@ApiModelProperty标注字段说明。

Unable to infer base url

简单三步,我们项目就集成了Swagger接口文档,赶紧启动服务,访问http://localhost:8080/swagger-ui.html体验一下。

好吧,出了点小问题,不过不用慌。

出现这个问题的原因是因为我们加上了ResponseBodyAdvice统一处理返回值/响应体,导致给Swagger的返回值也包装了一层,UI页面无法解析。可以通过http://localhost:8080/v2/api-docs?group=SwaggerDemo观察Swagger返回的json数据。

既然知道了问题原因那就很好解决了,我们只需要在ResponseBodyAdvice处理类中只转换我们自己项目的接口即可。

@RestControllerAdvice(basePackages = "com.jianzh5.blog")
@Slf4j
public class ResponseAdvice implements ResponseBodyAdvice<Object> {...
}

通过添加basePackage属性限定统一返回值的范围,这样就不影响Swagger了。

重启服务器再次访问swagger接口地址,就可以看到接口文档页面了。

For input string: ""

Swagger2.9.2有个bug,就是当我们参数实体有int类型的参数时,打开Swagger接口页面时后端会一直提示异常:

java.lang.NumberFormatException: For input string: ""at java.base/java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)at java.base/java.lang.Long.parseLong(Long.java:702)at java.base/java.lang.Long.valueOf(Long.java:1144)

有两种解决方案:

  1. 给int类型的字段使用@ApiModelPorperty注解时添加example属性

@ApiModelProperty(value = "年龄",example = "10")
private int age;
  1. 去除原swagger中的swagger-modelsswagger-annotations,自行引入高版本的annotations和models

<dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger2</artifactId><version>2.9.2</version><exclusions><exclusion><groupId>io.swagger</groupId><artifactId>swagger-annotations</artifactId></exclusion><exclusion><groupId>io.swagger</groupId><artifactId>swagger-models</artifactId></exclusion></exclusions>
</dependency><dependency><groupId>io.swagger</groupId><artifactId>swagger-annotations</artifactId><version>1.5.22</version>
</dependency>
<dependency><groupId>io.swagger</groupId><artifactId>swagger-models</artifactId><version>1.5.22</version>
</dependency>

集成Swagger过程中虽然会出现两个小问题,解决后我们就可以愉快享受Swagger给我们带来的便利了。

 

Swagger美化

Swagger原生UI有点丑,我们可以借助Swagger的增强工具knife4j优化一下。

第一步:引入依赖包

 <!--整合Knife4j-->
<dependency><groupId>com.github.xiaoymin</groupId><artifactId>knife4j-spring-boot-starter</artifactId><version>2.0.4</version>
</dependency>

由于knife4j中已经带了swagger-annotationsswagger-models的依赖,所以我们可以把上文中手动添加的两个依赖删除。

第二步:启用knife4j增强

@Configuration
@EnableSwagger2
@ConditionalOnClass(Docket.class)
@EnableKnife4j
public class SwaggerConfig {...
}

通过上面两步我们就完成了Swagger的美化,通过浏览器访问http://localhost:8080/doc.html即可看到效果。

 

Swagger参数分组

看到这里的同学心理肯定会想,就这?这就是老鸟的做法?跟我们新手也没啥区别呀!

别急,我们先来看一个效果。

首先我们定义了两个接口,一个新增,一个编辑

@ApiOperation("新增")
@PostMapping(value = "/valid/add")
public String add(@Validated(value = {ValidGroup.Crud.Create.class}) ValidVO validVO){log.info("validEntity is {}", validVO);return "test3 valid success";
}@ApiOperation("更新")
@PostMapping(value = "/valid/update")
public String update(@Validated(value = ValidGroup.Crud.Update.class) ValidVO validVO){log.info("validEntity is {}", validVO);return "test4 valid success";
}

注意看,这里用的是同一个实体ValidVO来接收前端参数,只不过使用了参数校验中的分组,然后我们打开kife4j页面观察两者的接口文档有何不同。

新增:

编辑:

通过上面可以看到,虽然用于接受参数的实体一样,但是当分组不一样时展示给前端的参数也不一样,这就是Swagger的分组功能。

当然原生的Swagger是不支持分组功能的,我们需要对Swagger进行扩展。我已经将代码上传到了github上,由于代码量比较多这里就不展示了,大家可以自行查阅。

引入扩展类后还需要在Swagger配置类SwaggerConfig中注入对应的Bean。
@Configuration
@EnableSwagger2
@ConditionalOnClass(Docket.class)
@EnableKnife4j
public class SwaggerConfig {...@Bean@Order(SwaggerPluginSupport.SWAGGER_PLUGIN_ORDER + 1000)public GroupOperationModelsProviderPlugin groupOperationModelsProviderPlugin() {return new GroupOperationModelsProviderPlugin();}@Bean@Order(SwaggerPluginSupport.SWAGGER_PLUGIN_ORDER + 1000)public GroupModelBuilderPlugin groupModelBuilderPlugin() {return new GroupModelBuilderPlugin();}@Bean@Order(SwaggerPluginSupport.SWAGGER_PLUGIN_ORDER + 1000)public GroupModelPropertyBuilderPlugin groupModelPropertyBuilderPlugin() {return new GroupModelPropertyBuilderPlugin();}@Bean@Order(SwaggerPluginSupport.SWAGGER_PLUGIN_ORDER + 1000)public GroupExpandedParameterBuilderPlugin groupExpandedParameterBuilderPlugin() {return new GroupExpandedParameterBuilderPlugin();}@Bean@Order(SwaggerPluginSupport.SWAGGER_PLUGIN_ORDER + 1000)public GroupOperationBuilderPlugin groupOperationBuilderPlugin() {return new GroupOperationBuilderPlugin();}@Bean@Primary@Order(SwaggerPluginSupport.SWAGGER_PLUGIN_ORDER + 1000)public GroupModelAttributeParameterExpander groupModelAttributeParameterExpander(FieldProvider fields, AccessorsProvider accessors, EnumTypeDeterminer enumTypeDeterminer) {return new GroupModelAttributeParameterExpander(fields, accessors, enumTypeDeterminer);}}

分组使用说明

1.在bean对象的属性里配置如下注释

@Null(groups = ValidGroup.Crud.Create.class)
@NotNull(groups = ValidGroup.Crud.Update.class,message = "应用ID不能为空")
@ApiModelProperty(value = "应用ID",example = "cloud")
private String appId;

当新增场景的时候,appId为空,不需要传值;当修改场景的时候,appId不能为空,需要传值 ;其他没有配置组的皆为默认组(Default)

2.在接口参数的时候加入组规则校验

 @ApiOperation("新增")@PostMapping(value = "/valid/add")public String add(@Validated(value = {ValidGroup.Crud.Create.class}) ValidVO validVO){log.info("validEntity is {}", validVO);return "test3 valid success";}

当前接口会针对默认组的bean属性进行校验,同时针对保存常见的属性进行校验。

 

小结

Swagger集成相对来说还是很简单的,虽然在集成过程中也出现了几个小问题,不过也很容易就解决了。今天文章的重点内容是Swagger分组功能,跟之前的参数校验文章一样,很多同学遇到这种分组场景时往往会选择创建多个实体类,虽然也能解决问题,只不过总是有点别扭。

不过遗憾的是,本文中Swagger的分组扩展只支持Swagger2,至于新版本Swagger3就不怎么友好了。

有道无术,术可成;有术无道,止于术

欢迎大家关注Java之道公众号

好文章,我在看❤️

SpringBoot 如何生成接口文档,老鸟们都这么玩的!相关推荐

  1. SpringBoot自动生成接口文档

    跟大家介绍一个自动生成接口文档的工具包,作者的理念是注释即文档,在写代码的时候写上注释,项目启动后就会生成接口文档,非常方便,省去了Swagger写注解的过程. 仓库地址:https://github ...

  2. SpringBoot 整合Smart-doc生成接口文档

    之前我在SpringBoot老鸟系列中专门花了大量的篇幅详细介绍如何集成Swagger,以及如何对Swagger进行扩展让其支持接口参数分组功能.详情可见:SpringBoot 如何生成接口文档,老鸟 ...

  3. 实体类 接口_spring-boot-route(五)整合Swagger生成接口文档

    目前,大多数公司都采用了前后端分离的开发模式,为了解决前后端人员的沟通问题,后端人员在开发接口的时候会选择使用swagger2来生成对应的接口文档,swagger2提供了强大的页面调试功能,这样可以有 ...

  4. 自动生成接口文档之JApiDocs教程

    JApiDocs教程 前言 作为一名优秀的程序员来说,由于涉及到要与前端进行对接,所以避免不了的就是写接口文档.写完接口文档,一旦代码返回结果,参数等出现变动,接口文档还得随之改动,十分麻烦,违背了我 ...

  5. 通过Swagger快速生成接口文档

    1.添加依赖: 在pom中添加swagger的依赖,这里使用的swagger版本为2.7.0,springboot相应版本为2.0.3,jdk1.8 示例如下: <swagger.version ...

  6. SpringBoot 自动生成API文档

    SpringBoot 自动生成API文档 在做项目的时候,如果项目是前后分离的,后端一定要和前端或者是移动端对接接口,那么问题来了,接口是不是要自己写给他们看,一般的会采用Excel或者Word来写, ...

  7. RESTful之自动生成接口文档

    REST framework可以自动帮助我们生成接口文档.[必须是继承自rest_framework才会自动生成文档] 接口文档以网页的方式呈现. 自动接口文档能生成的是继承自APIView及其子类的 ...

  8. Django框架深入了解_05 (Django中的缓存、Django解决跨域流程(非简单请求,简单请求)、自动生成接口文档)(二)

    二.跨域: 回到顶部 跨域知识介绍: 点我以前博客 跨域解决方法:CORS:跨域资源共享 CORS请求分类(简单请求和非简单请求) 简单请求(simple request):只需要在头信息之中增加一个 ...

  9. Django框架深入了解_05 (Django中的缓存、Django解决跨域流程(非简单请求,简单请求)、自动生成接口文档)(一)

    阅读目录 一.Django中的缓存: 前戏: Django中的几种缓存方式: Django中的缓存应用: 二.跨域: 跨域知识介绍: CORS请求分类(简单请求和非简单请求) 示例: 三.自动生成接口 ...

最新文章

  1. 华人团队用Transformer做风格迁移,速度快、可试玩,网友却不买账
  2. 21.3 共享受限的资源
  3. 单片机led闪烁代码_单片机驱动LED发光二极管的电路以及编程
  4. linux删除文件_Linux删除文件夹用什么命令
  5. mysql 1.42_MySQL索引(一)
  6. 查找——顺序、二分法、斐波那契、插值、分块
  7. 挑战安卓会死?华为鸿蒙正为国产操作系统杀出一条路 | 涛滔不绝
  8. async spring 默认线程池_springboot中@Async默认线程池导致OOM问题
  9. 阶段1 语言基础+高级_1-3-Java语言高级_07-网络编程_第1节 网络通信概述_4_IP地址...
  10. 黑客是如何监视女友出轨信息的,痛心的经历!
  11. 什么耳机适合华为手机?通话质量好的蓝牙耳机
  12. Keras框架使用Vnet2d模型对遥感图像语义分割
  13. 妖精为什么吃不到唐僧肉
  14. 杰理之设置立体声输出,播左右声道歌曲后DAC没有声音或声音变小
  15. android tf卡及u盘_android8.1系统修改第三方app无法读写U盘或者内部SD卡的问题
  16. Java——nefu
  17. linux下怎么批量命名文件,linux下的文件操作——批量重命名
  18. rust的所有权与引用
  19. Excel工具箱-方方格子
  20. 树莓派LED驱动编写

热门文章

  1. 地图定点图表联动_拿下这套地图组件,快人一步做出炫酷报表!
  2. html自动年份版权,如何将html5日期输入限制在合理的年份
  3. oracle备份信息在控制文件丢失,恢复之利用备份在所有控制文件丢失情况下恢复(一)...
  4. java声明公共构造函数_确保控制器具有无参数的公共构造函数错误
  5. java算法提高求最大值_藍橋杯 算法提高 求最大值
  6. 计算机考研310分什么水平,知乎工学考研310是什么水平
  7. 计算机网络之数据链路层:12、CSMA/CA协议-随机访问介质访问控制
  8. 计算机网络之数据链路层:8、介质访问控制之信道划分介质访问控制
  9. (计算机组成原理)第一章计算机系统概述-王道重点习题及杂项总结
  10. (计算机组成原理)第三章存储系统-第七节2:页式/段式虚拟存储器