swagger

1.了解

  1. 简介

你可能尝试过写完一个接口后,自己去创建接口文档,或者修改接口后修改接口文档。多 了之后,你肯定会发生一个操作,那就是忘记了修改文档或者创建文档(除非你们公司把接口 文档和写接口要求得很紧密😓忘记写文档就扣工资?,否则两个分离的工作总是有可能遗漏 的)。而swagger就是一个在你写接口的时候自动帮你生成接口文档的东西,只要你遵循它的 规范并写一些接口的说明注解即可

  1. 优点

1.自动生成文档,只需要在接口中使用注解进行标注,就能生成对应的接口文档。 2.自动更新文档,由于是动态生成的,所以如果你修改了接口,文档也会自动对应修改(如果你也更新了注解的话)。这样就不会发送我修改了接口,却忘记更新接口文档的情况。 3.支持在线调试,swagger提供了在线调用接口的功能。

  1. 缺点

1.不能创建测试用例,所以他暂时不能帮你处理完所有的事情。他只能提供一个简单的在线调试,如果你想存储你的测试用例,可以使用Postman或者YAPI这样支持创建测试用户的功能。 2.要遵循一些规范,它不是任意规范的。比如说,你可能会返回一个json数据,而这个数据可能是一个Map格式的,那么我们此时不能标注这个Map格式的返回数据的每个字段的说明,而如果它是一个实体类的话,我们可以通过标注类的属性来给返回字段加说明。也比如说,对于swagger,不推荐在使用GET方式提交数据的时候还使用Body,仅推荐使用query参数、header参数或者路径参数,当然了这个限制只适用于在线调试。 3.没有接口文档更新管理,虽然一个接口更新之后,可能不会关心旧版的接口信息,但你“可能”想看看旧版的接口信息,例如有些灰度更新发布的时候可能还会关心旧版的接口。那么此时只能由后端去看看有没有注释留下了,所以可以考虑接口文档大更新的时候注释旧版的,然后写下新版的。【当然这个问题可以通过导出接口文档来对比。】 4.虽然现在Java的实体类中有不少模型,po,dto,vo等,模型的区分是为了屏蔽一些多余参数,比如一个用户登录的时候只需要username,password,但查权限的时候需要连接上权限表的信息,而如果上述两个操作都是使用了User这个实体的话,在文档中就会自动生成了多余的信息,这就要求了你基于模型来创建多个实体类,比如登录的时候一个LoginForm,需要用户-权限等信息的时候才使用User类。(当然了,这个问题等你会swagger之后你就大概就会怎么规避这个问题了。)

2.springboot集成swagger

  1. 引jar包

            <dependency>
               <groupId>io.springfox</groupId>
               <artifactId>springfox-swagger2</artifactId>
               <version>2.9.2</version>
           </dependency>
           <dependency>
               <groupId>io.springfox</groupId>
               <artifactId>springfox-swagger-ui</artifactId>
               <version>2.9.2</version>
           </dependency>
  2. 配置swagger

    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import springfox.documentation.builders.ApiInfoBuilder;
    import springfox.documentation.builders.PathSelectors;
    import springfox.documentation.builders.RequestHandlerSelectors;
    import springfox.documentation.service.ApiInfo;
    import springfox.documentation.spi.DocumentationType;
    import springfox.documentation.spring.web.plugins.Docket;
    import springfox.documentation.swagger2.annotations.EnableSwagger2;

    // 标明是配置类
    @Configuration
    //开启swagger功能
    @EnableSwagger2
    public class SwaggerConfig {
       @Bean
       public Docket createRestApi() {
           // DocumentationType.SWAGGER_2 固定的,代表swagger2
           return new Docket(DocumentationType.SWAGGER_2)
                   // 如果配置多个文档的时候,那么需要配置groupName来分组标识
                   //.groupName("分布式任务系统")
                   // 用于生成API信息
                  .apiInfo(apiInfo())
                   // select()函数返回一个ApiSelectorBuilder实例,用来控制接口被swagger做成文档
                  .select()
                   // 用于指定扫描哪个包下的接口
                  .apis(RequestHandlerSelectors.basePackage("com.zhuoyue.controller"))
                   // 选择所有的API,如果你想只为部分API生成文档,可以配置这里
                  .paths(PathSelectors.any())
                  .build();
      }

       /**
        * 用于定义API主界面的信息,比如可以声明所有的API的总标题、描述、版本
        * @return 所有的API的总标题、描述、版本
        */
       private ApiInfo apiInfo() {
           return new ApiInfoBuilder()
                   // 可以用来自定义API的主标题
                  .title("XX项目API")
                   // 可以用来描述整体的API
                  .description("XX项目SwaggerAPI管理")
                   // 用于定义服务的域名
                  .termsOfServiceUrl("")
                   // 可以用来定义版本。
                  .version("1.0")
                  .build();
      }
    }

3.测试

http://localhost:8080/swagger-ui.html

4.swagger相关注解

  1. 自定义controller接口组(类)@Api(tags = "用户管理")

@Api(tags = "用户管理")
@RestController
public class UserController {
   // 注意,对于swagger,不要使用@RequestMapping,
   // 因为@RequestMapping支持任意请求方式,swagger会为这个接口生成7种请求方式的接口文档
   @GetMapping("/info")
   public String info(String id){
       return "aaa";
  }
}
  1. 描述接口(方法)@ApiOperation

    @ApiOperation(value = "用户测试",notes = "用户测试notes")
   @GetMapping("/test")
   public String test(String id){
       return "test";
  }

value:可以当作是接口的简称 notes:接口的描述 tags:可以额外定义接口组,比如这个接口外层已经有@Api(tags = "用户管理"),将接口划分到了“用户管理”中,但你可以额外的使用tags,例如tags = "角色管理"让角色管理中也有这个接口文档。

  1. 实体类描述@ApiModel

  1. 实体类属性描述@ApiModelProperty

// 先使用@ApiModel来标注类
@ApiModel(value="用户登录表单对象",description="用户登录表单对象")
public class LoginForm {
   // 使用ApiModelProperty来标注字段属性。
   @ApiModelProperty(value = "用户名",required = true,example = "root")
   private String username;
   @ApiModelProperty(value = "密码",required = true,example = "123456")
   private String password;

   // 此处省略入参赋值时需要的getter,setter,swagger也需要这个
}
  1. 普通参数@ApiImplicitParams和@ApiImplicitParam

    对于GET方式,swagger不推荐使用body方式来传递数据,所以虽然Spring MVC可以自动封装参数,但对于GET请求还是不要使用form-data,json等方式传递参数,除非你使用Postman来测试接口,swagger在线测试是不支持这个操作的) 对于非实体类参数,可以使用@ApiImplicitParams和@ApiImplicitParam来声明请求参数。 @ApiImplicitParams用在方法头上,@ApiImplicitParam定义在@ApiImplicitParams里面,一个@ApiImplicitParam对应一个参数。 @ApiImplicitParam常用配置项:

    name:用来定义参数的名字,也就是字段的名字,可以与接口的入参名对应。如果不对应,也会生成,所以可以用来定义额外参数! value:用来描述参数 required:用来标注参数是否必填 paramType有path,query,body,form,header等方式,但对于对于非实体类参数的时候,常用的只有path,query,header;body和form是不常用的。body不适用于多个零散参数的情况,只适用于json对象等情况。【如果你的接口是form-data,x-www-form-urlencoded的时候可能不能使用swagger页面API调试,但可以在后面讲到基于BootstrapUI的swagger增强中调试,基于BootstrapUI的swagger支持指定form-data或x-www-form-urlencoded】

1.声明入参是URL参数
       // 使用URL query参数
   @ApiOperation(value = "登录接口2",notes = "登录接口的说明2")
   @ApiImplicitParams({
           @ApiImplicitParam(name = "username",//参数名字
                   value = "用户名",//参数的描述
                   required = true,//是否必须传入
                   //paramType定义参数传递类型:有path,query,body,form,header
                   paramType = "query"
                  )
          ,
           @ApiImplicitParam(name = "password",//参数名字
                   value = "密码",//参数的描述
                   required = true,//是否必须传入
                   paramType = "query"
                  )
  })
   @PostMapping(value = "/login2")
   public LoginForm login2(String username,String password){
       System.out.println(username+":"+password);
       LoginForm loginForm = new LoginForm();
       loginForm.setUsername(username);
       loginForm.setPassword(password);
       return loginForm;
  }
2.声明入参是URL路径参数
       // 使用路径参数
   @PostMapping("/login3/{id1}/{id2}")
   @ApiOperation(value = "登录接口3",notes = "登录接口的说明3")
   @ApiImplicitParams({
           @ApiImplicitParam(name = "id1",//参数名字
                   value = "用户名",//参数的描述
                   required = true,//是否必须传入
                   //paramType定义参数传递类型:有path,query,body,form,header
                   paramType = "path"
          )
          ,
           @ApiImplicitParam(name = "id2",//参数名字
                   value = "密码",//参数的描述
                   required = true,//是否必须传入
                   paramType = "path"
          )
  })
   public String login3(@PathVariable Integer id1,@PathVariable Integer id2){
       return id1+":"+id2;
  }
3.声明入参是header参数
       // 用header传递参数
   @PostMapping("/login4")
   @ApiOperation(value = "登录接口4",notes = "登录接口的说明4")
   @ApiImplicitParams({
           @ApiImplicitParam(name = "username",//参数名字
                   value = "用户名",//参数的描述
                   required = true,//是否必须传入
                   //paramType定义参数传递类型:有path,query,body,form,header
                   paramType = "header"
          )
          ,
           @ApiImplicitParam(name = "password",//参数名字
                   value = "密码",//参数的描述
                   required = true,//是否必须传入
                   paramType = "header"
          )
  })
   public String login4( @RequestHeader String username,
                         @RequestHeader String password){
       return username+":"+password;
  }
4.声明文件上传参数
       // 有文件上传时要用@ApiParam,用法基本与@ApiImplicitParam一样,不过@ApiParam用在参数上
   // 或者你也可以不注解,swagger会自动生成说明
   @ApiOperation(value = "上传文件",notes = "上传文件")
   @PostMapping(value = "/upload")
   public String upload(@ApiParam(value = "图片文件", required = true)MultipartFile uploadFile){
       String originalFilename = uploadFile.getOriginalFilename();

       return originalFilename;
  }
   

   // 多个文件上传时,**swagger只能测试单文件上传**
   @ApiOperation(value = "上传多个文件",notes = "上传多个文件")
   @PostMapping(value = "/upload2",consumes = "multipart/*", headers = "content-type=multipart/form-data")
   public String upload2(@ApiParam(value = "图片文件", required = true,allowMultiple = true)MultipartFile[] uploadFile){
       StringBuffer sb = new StringBuffer();
       for (int i = 0; i < uploadFile.length; i++) {
           System.out.println(uploadFile[i].getOriginalFilename());
           sb.append(uploadFile[i].getOriginalFilename());
           sb.append(",");
      }
       return sb.toString();
  }



   // 既有文件,又有参数
   @ApiOperation(value = "既有文件,又有参数",notes = "既有文件,又有参数")
   @PostMapping(value = "/upload3")
   @ApiImplicitParams({
           @ApiImplicitParam(name = "name",
                   value = "图片新名字",
                   required = true
          )
  })
   public String upload3(@ApiParam(value = "图片文件", required = true)MultipartFile uploadFile,
                         String name){
       String originalFilename = uploadFile.getOriginalFilename();

       return originalFilename+":"+name;
  }
  1. 响应为非实体类@ApiResponses和@ApiResponse

    // 其他类型的,此时不能增加字段注释,所以其实swagger推荐使用实体类
   @ApiOperation(value = "非实体类",notes = "非实体类")
   @ApiResponses({
           @ApiResponse(code=200,message = "调用成功"),
           @ApiResponse(code=401,message = "无权限" )
  }
  )
   @PostMapping("/role2")
   public String role2(){
       return " {\n" +
               " name:\"广东\",\n" +
               "     citys:{\n" +
               "         city:[\"广州\",\"深圳\",\"珠海\"]\n" +
               "     }\n" +
               " }";
  }

5.Swagger UI增强

swagger-bootstrap-ui

  1. 引jar包

<!-- 引入swagger-bootstrap-ui依赖包-->
       <dependency>
           <groupId>com.github.xiaoymin</groupId>
           <artifactId>swagger-bootstrap-ui</artifactId>
           <version>1.8.7</version>
       </dependency>
  1. 写配置

@Configuration // 标明是配置类
@EnableSwagger2 //开启swagger功能
@EnableSwaggerBootstrapUI // 开启SwaggerBootstrapUI
public class SwaggerConfig {
   // 省略配置内容
}
  1. 测试地址http://localhost:8080/doc.html