【SpringCloud】SpringBoot集成Swagger 常用Swagger注解

概述:SpringBoot集成Swagger 常用Swagger注解

导语

相信无论是前端还是后端开发,都或多或少地被接口文档折磨过。前端经常抱怨后端给的接口文档与实际情况不一致。后端又觉得编写及维护接口文档会耗费不少精力,经常来不及更新。其实无论是前端调用后端,还是后端调用后端,都期望有一个好的接口文档。但是这个接口文档对于程序员来说,就跟注释一样,经常会抱怨别人写的代码没有写注释,然而自己写起代码起来,最讨厌的,也是写注释。所以仅仅只通过强制来规范大家是不够的,随着时间推移,版本迭代,接口文档往往很容易就跟不上代码了

Swagger是什么?它能干什么?

官网地址:https://swagger.io/

发现了痛点就要去找解决方案。解决方案用的人多了,就成了标准的规范,这就是Swagger 的由来。

通过这套规范,你只需要按照它的规范去定义接口及接口相关的信息。再通过Swagger 衍生出来的一系列项目和工具,就可以做到生成各种格式的接口文档 , 生成多种语言的客户端和服务端的代码,以及在线接口调试页面等等 。这样,如果按照新的开发模式,在开发新版本或者迭代版本的时候,只需要更新Swagger描述文件,就可以自动生成接口文档和客户端服务端代码,做到调用端代码、服务端代码以及接口文档的一致性。

框架说明

现在SWAGGER官网主要提供了几种开源工具,提供相应的功能。可以通过配置甚至是修改源码以达到你想要的效果:

作用:

  1. 接口的文档在线自动生成
  2. 功能测试

Spring集成Swagger

初始实现步骤

添加Maven依赖
 <!--        bootstrap-ui   访问http://localhost:8080/doc.html--><dependency><groupId>com.github.xiaoymin</groupId><artifactId>swagger-bootstrap-ui</artifactId><version>1.9.1</version></dependency><!--        swagger依赖 访问http://localhost:8080/swagger-ui.html--><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>
编写HelloController,测试确保运行成功
package com.stringzhua.springcloud_swagger01.controller;import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;/*** @Author Stringzhua* @Date 2024/9/30 11:41* description:*/
@RestController
public class HelloController {@RequestMapping("/hello")public String hello() {return "hello swagger";}
}
编写一个配置类-SwaggerConfig来配置 Swagger
@Configuration//配置类
@EnableSwagger2// 开启Swagger2的自动配置
public class SwaggerConfig {}
访问测试 :http://localhost:8080/swagger-ui.html ,可以看到swagger的界面;

如果启动报错空指针是因为springboot2.6.0中将SpringMVC 默认路径匹配策略从AntPathMatcher 更 改为PathPatternParser,导致出错

可以在启动类上加上@EnableWebMvc注解或者在配置中切换为原先的AntPathMatcher(加注解不太行)

spring.mvc.pathmatch.matching-strategy=ant_path_matcher

配置Swagger

Swagger实例Bean是Docket,所以通过配置Docket实例来配置Swagger
@Bean //配置docket以配置Swagger具体参数public Docket docket() {return new Docket(DocumentationType.SWAGGER_2);}
可以通过apiInfo()属性配置文档信息
 //配置文档信息private ApiInfo apiInfo() {Contact contact = new Contact("联系人名字", "http://xxx.xxx.com/联系人访问链接", "联系人邮箱");return new ApiInfo("Swagger学习", // 标题"学习演示如何配置Swagger", // 描述"v2.0", // 版本"http://zkt .com", // 组织链接contact, // 联系人信息"Apach 2.0 许可", // 许可"许可链接", // 许可连接new ArrayList<>()// 扩展);}
Docket 实例关联上 apiInfo()
 @Bean //配置docket以配置Swagger具体参数public Docket docket() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo());}
重启项目,访问测试 http://localhost:8080/swagger-ui.html 看下效果;

配置扫描接口

构建Docket时通过select()方法配置怎么扫描接口
 @Bean //配置docket以配置Swagger具体参数public Docket docket() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo()).groupName("hello") // 配置分组.select()// 通过.select()方法,去配置扫描接口,RequestHandlerSelectors配置如何扫描接口.apis(RequestHandlerSelectors.basePackage("com.zkt.springboot_swagger01.controller")).paths(PathSelectors.any()).build();}
重启项目测试,由于我们配置根据包的路径扫描接口,所以我们只能看到一个类

除了通过包路径配置扫描接口外,还可以通过配置其他方式扫描接口,这里注释一下所有的配置方式
any() // 扫描所有,项目中的所有接口都会被扫描到
none() // 不扫描接口
// 通过方法上的注解扫描,如withMethodAnnotation(GetMapping.class)只扫描get请求
withMethodAnnotation(final Class<? extends Annotation> annotation)
// 通过类上的注解扫描,如.withClassAnnotation(Controller.class)只扫描有controller注解
的类中的接口
withClassAnnotation(final Class<? extends Annotation> annotation)
basePackage(final String basePackage) // 根据包路径扫描接口
除此之外,我们还可以配置接口扫描过滤
@Bean
public Docket docket() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo()).select() // 通过.select()方法,去配置扫描接口, RequestHandlerSelectors配置如何扫// 描接口.apis(RequestHandlerSelectors.basePackage("com.zkt.swagger.controller"))// 配置如何通过path过滤,即这里只扫描请求以/zkt开头的接口.paths(PathSelectors.ant("/zkt/**")).build();
}
这里的可选值还有
any() // 任何请求都扫描
none() // 任何请求都不扫描
regex(final String pathRegex) // 通过正则表达式控制
ant(final String antPattern) // 通过ant()控制

配置API分组

  1. 如果没有配置分组,默认是default。通过groupName()方法即可配置分组
  2. 重启项目查看分组
  3. 如何配置多个分组?配置多个分组只需要配置多个docket即可:
@Bean
public Docket docket1(){return new Docket(DocumentationType.SWAGGER_2).groupName("group1");
}
@Bean
public Docket docket2(){return new Docket(DocumentationType.SWAGGER_2).groupName("group2");
}
@Bean
public Docket docket3(){return new Docket(DocumentationType.SWAGGER_2).groupName("group3");
}

拓展:其他皮肤

我们可以导入不同的包实现不同的皮肤定义

  1. 默认的访问 http://localhost:8080/swagger-ui.html
<dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger-ui</artifactId><version>2.9.2</version>
</dependency>
  1. bootstrap-ui 访问 ****http://localhost:8080/doc.html

常用Swagger注解

Swagger通过注解生成接口文档,包括接口名、请求方法、参数、返回信息等等。

@Api

修饰整个类,描述Controller的作用

语法:@Api(tag=“类的作用,可以在UI界面上看到的注释”, value=“/类的访问路径”, description=“类的文字描述”)

@ApiOperation

描述一个类的一个方法,说明方法的作用

语法:@ApiOperation(value=“接口名称”, httpMethod=“接口请求方式”, response=“接口返回 参数类型”, notes=“接口发布说明”)

@ApiImplicitParam

一个请求参数

语法:@ApiImplicitParam(required=“是否必须参数”, name=“参数名称”, value=“参数具体描述”, dataType=“变量类型”, paramType=“请求方式” )

header -> 请求参数的获取:@RequestHeader

query -> 请求参数的获取:@RequestParam

path(用于restful接口)-> 请求参数的获取:@PathVariable

body(不常用) -> 请求参数的获取:@RequestBody

form(不常用) -> 请求参数的获取:@RequestParam

@ApiImplicitParams

多个请求参数

@ApiParam

单个参数描述

语法:@ApiParam(required=“是否必须参数”, name=“参数名称”, value=“参数具体描述”,dataType=“变量类型”,paramType=“请求方式”)

@ApiResponse

HTTP响应应答描述

语法:@ApiResponse(code=400, message=“Invalid user supplied”)

@ApiResponses

HTTP响应整体描述

语法:@ApiResponses({@ApiResponse(code=400, message=“Invalid Order”)})

@ApiModel

用对象实体类作为入参

@ApiModelProperty

用对象接收参数时,描述对象的一个字段

@ApiIgnore

使用该注解忽略这个API

@ApiError

发生错误返回的信息

演示

编码:

POJO层:

Admin.java

package com.stringzhua.springboot_swagger02.pojo;import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;import java.sql.Date;@Data
@NoArgsConstructor
@AllArgsConstructor
@ApiModel(value = "管理员实体列",description = "管理员实体类")
public class Admin {@ApiModelProperty(name = "adminId",value = "管理员编号",example = "001")private int adminId;//管理员编号@ApiModelProperty(name = "adminLoginAccount",value = "管理员登录账号",example = "13991267980")private String adminLoginAccount;//管理员登录账号(手机号码)@ApiModelProperty(name = "adminLoginPassword",value = "登录登录密码",example = "123456")private String adminLoginPassword;//登录登录密码(默认为手机号码后8位)@ApiModelProperty(name = "adminRole",value = "管理员角色",example = "管理员")private String adminRole;//管理员角色(超级管理员或普通管理员)@ApiModelProperty(name = "adminLastLoginTime",value = "最后登录时间")private Date adminLastLoginTime;//最后登录时间(2024年09月30日 15:30:40)
}

Result.java

package com.stringzhua.springboot_swagger02.pojo;import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;//结果集
@Data
@NoArgsConstructor
@AllArgsConstructor
@ApiModel(value = "控制器方法返回值",description = "自定义结果集")
public class Result {@ApiModelProperty(name = "falge",value = "结果集状态",example = "true")private boolean falge;//状态@ApiModelProperty(name = "message",value = "提示信息",example = "查询成功")private String message;//提示信息@ApiModelProperty(name = "data",value = "返回数据")private Object data;//返回数据}

Config层:

SwaggerConfig.java

package com.stringzhua.springboot_swagger02.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import springfox.documentation.builders.PathSelectors;
import springfox.documentation.builders.RequestHandlerSelectors;
import springfox.documentation.service.ApiInfo;
import springfox.documentation.service.Contact;
import springfox.documentation.spi.DocumentationType;
import springfox.documentation.spring.web.plugins.Docket;import java.util.ArrayList;
/*** @Author Stringzhua* @Date 2024/9/30 11:41* description:*/
@Configuration
public class SwaggerConfig {@Beanpublic Docket docket() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo()).select().apis(RequestHandlerSelectors.basePackage("com.stringzhua.springboot_swagger02.controller"))//扫描路径.paths(PathSelectors.any())//定义哪些路径的接口需要生成文档.build();}//配置文档的信息private ApiInfo apiInfo() {Contact contact = new Contact("项目负责人","https://gitee.com/jun-0912/stringzhua-takeout","2785649457@qq.com");return new ApiInfo("Swagger2.0文档学习技能点",//标题"学习配置Swagger",//描述"v3.0",//版本"https://gitee.com/jun-0912",//组织链接contact,//联系人信息"Apach 2.0",//许可"https://gitee.com/jun-0912/sql-mother",//许可链接new ArrayList<>()//拓展);}
}

Controller层:

UserController.java

package com.stringzhua.springboot_swagger02.controller;import com.stringzhua.springboot_swagger02.pojo.Admin;
import com.stringzhua.springboot_swagger02.pojo.Result;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import org.springframework.web.bind.annotation.*;import java.sql.Date;/*** @Author Stringzhua* @Date 2024/9/30 11:41* description:*/
@RestController
@RequestMapping("/user")
@Api(tags = "UserController|控制器1测试swagger", value = "/user")
public class UserController {/************************************修饰参数****************************************/@GetMapping(value = "/findByName")@ApiOperation(value = "根据用户名查询")public Result findByNamw(@RequestParam String userName) {boolean b = userName.equals("一猫人");return b == true ? new Result(true, "查询成功", "大熊猫本猫") : new Result(false, "查询失败", null);}@GetMapping(value = "/findByNameAndSex")@ApiOperation(value = "根据用户名与性别查询", notes = "我说这是一个描述你信吗?这是可以省略的")public Result findByNameAndSex(String userName, String Sex) {if (userName.equals("一猫人") && Sex.equals("女")) {return new Result(true, "查询成功", "大熊猫本猫");} else {return new Result(false, "查询失败", null);}}@GetMapping(value = "/findByAge/{age}")@ApiOperation(value = "根据用户的年龄查询", notes = "我说这是一个描述你信吗?这是可以省略的")public Result findByAge(@PathVariable("age") Integer userAge) {if (userAge > 18) {return new Result(true, "查询成功", "大熊猫本猫已成年");} else {return new Result(false, "查询失败", null);}}@PostMapping(value = "/saveAdmin1")@ApiOperation(value = "新增管理员信息1", notes = "我说这是一个描述你信吗?这是可以省略的")public Result saveAdmin1(Admin admin) {if (admin.getAdminId() > 10) {return new Result(true, "查询成功", "大熊猫本猫已成年");} else {return new Result(false, "查询失败", null);}}@PostMapping(value = "/saveAdmin2")@ApiOperation(value = "新增管理员信息2", notes = "我说这是一个描述你信吗?这是可以省略的")public Result saveAdmin2(@RequestBody Admin admin) {if (admin.getAdminId() > 10) {return new Result(true, "查询成功", "大熊猫本猫已成年");} else {return new Result(false, "查询失败", null);}}/************************************返回值****************************************/@DeleteMapping(value = "/deleteById/{id}")@ApiOperation(value = "根据用户编号删除信息", notes = "")public Admin deleteById(@PathVariable("id") Integer id) {return new Admin(1, "一猫人", "管理员", "普通管理员", new Date(System.currentTimeMillis()));}
}

启动类:

package com.stringzhua.springboot_swagger02;import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import springfox.documentation.swagger2.annotations.EnableSwagger2;@SpringBootApplication
@EnableSwagger2
public class SpringbootSwagger02Application {public static void main(String[] args) {SpringApplication.run(SpringbootSwagger02Application.class, args);}}

配置文件:

spring.application.name=springboot_swagger02
spring.mvc.pathmatch.matching-strategy=ant_path_matcher

测试:

测试 /user/saveAdmin1
新增管理员信息1

点击try it out!

测试成功!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/467916.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

革命性AI搜索引擎!ChatGPT最新功能发布,无广告更智能!

文章目录 零、前言一、ChatGPT最新AI搜索引擎功能操作指导实战1:搜索新闻实战2:搜索天气实战3:搜索体育消息 二、感受 零、前言 大人&#xff0c;时代变了。 最强 AI 助力下的无广告搜索引擎终于问世。我们期待已久的这一刻终于到来了&#xff0c;从今天起&#xff0c;ChatGPT…

基于 CMSIS-PACK 移植Bootloader

基于 CMSIS-PACK 移植 1.准备工作 准备一份基础的裸机源码 (可通过 STM32CubeMx 可视化软件创建也可按照工程项目所需文档手动创建) 工程&#xff0c;如一份 stm32 包含一个支持 printf 的串口初始化代码。 2.安装Pack包 在 MDK 中部署 **MicroBoot **的第一步是获取对应的…

苍穹外卖day09超出配送范围前端不提示问题

同学们在写苍穹外卖项目day09时调用了百度地图api来判断用户地址是否超出配送范围&#xff0c; 但是在黑马官方的课程或资料中&#xff0c;出现这样的问题时只会向用户端的控制台报错并不会提醒用户 如下图&#xff1a; 解决方法&#xff1a; 其实解决方法很简单只需要找到向…

嵌入式linux中PWM控制与实现

大家好,今天主要给大家分享一下,如何使用linux系统里面的PWM的功能,可以控制对应电机的转速。 第一:PWM驱动基本简介 PWM就是脉冲宽度调制。 PWM信号有两个关键术语:频率和占空比,频率指的是开关的速度。占空比就是一个周期内高电平和低电平时间的比例,一个周期内高电…

CUDA系统学习之一软件堆栈架构

一、CPU与GPU体系架构 计算单元分布 CPU: 少量强大的ALU(算术逻辑单元)&#xff0c;通常4-8个核心GPU: 大量小型ALU&#xff0c;成百上千个计算核心特点&#xff1a;GPU更适合并行计算&#xff0c;可以同时处理大量数据控制单元(Control) CPU: 较大的控制单元&#xff0c;复杂的…

「QT」几何数据类 之 QPoint 整型点类

✨博客主页何曾参静谧的博客&#x1f4cc;文章专栏「QT」QT5程序设计&#x1f4da;全部专栏「VS」Visual Studio「C/C」C/C程序设计「UG/NX」BlockUI集合「Win」Windows程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「PK」Parasolid…

0x00基础算法 -- 0x01 位运算

资料来源&#xff1a;算法竞赛进阶指南活动 - AcWing 1、进制表示 二进制表示&#xff1a;m位二进制中&#xff0c;通常称最低位为第0位&#xff0c;从右到左以此类推&#xff0c;最高位为第m-1位。 常用十六进制表示的数字&#xff1a; 32位补码int&#xff08;十进制&#xf…

H5移动端预览PDF方法

新建页面 新建一个页面以便去预览对应的pdf 新建完后在 pages.json 文件内去新增对应路由 页面内容 <template><view class"page"><view class"pdf"><view id"demo"></view></view><view class"b…

嵌入式开发之线程

进程 vs 线程 进程在切换时系统开销大很多操作系统引入了轻量级进程LWP同一进程中的线程共享相同地址空间Linux不区分进程、线程(都会创建:task_strcut)线程特点: 通常线程指的是共享相同的地址空间的多个任务,使用多线程的好处 大大提高了任务切换的效率避免了额外的TLB…

【SQL实验】更新操作

完整代码在文章末尾【代码是自己的解答&#xff0c;并非标准答案&#xff0c;也有可能写错&#xff0c;文中可能会有不准确或待完善之处&#xff0c;恳请各位读者不吝批评指正&#xff0c;共同促进学习交流】 将素材“图书管理”文件下载到本地&#xff0c;并将其还原到SQL SER…

Hadoop(HDFS)

Hadoop是一个开源的分布式系统架构&#xff0c;旨在解决海量数据的存储和计算问题&#xff0c;Hadoop的核心组件包括Hadoop分布式文件系统&#xff08;HDFS&#xff09;、MapReduce编程模型和YARN资源管理器,最近需求需要用到HDFS和YARN。 文章目录 HDFS优缺点HDFS的读写原理 常…

Spire.PDF for .NET【页面设置】演示:获取 PDF 文件中的页数

计算 PDF 文件中的页数对于各种目的都至关重要&#xff0c;例如确定文档长度、组织内容和评估打印要求。除了使用 PDF 查看器了解页数信息外&#xff0c;您还可以通过编程自动执行该任务。在本文中&#xff0c;您将学习如何使用C#通过Spire.PDF for .NET获取 PDF 文件中的页数。…

stm32不小心把SWD和JTAG都给关了,程序下载不进去,怎么办?

因为想用STM32F103的PA15引脚&#xff0c;调试程序的时候不小心把SWD和JTAD接口都给关了&#xff0c;先看下罪魁祸首 GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);//关掉JTAG&#xff0c;不关SWGPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable, ENABLE);//关掉SW&am…

vue3使用element-plus,树组件el-tree增加引导线

vue3使用element-plus&#xff0c;树组件el-tree增加引导线 vue3项目element-plus&#xff0c;树组件el-tree增加引导线 element-plus组件库的el-tree样式 因为element的样式不满足当前的的需求&#xff0c;UI图&#xff0c;所以对el-tree进行增加了引导线 修改样式如下&am…

pytest简单使用

一&#xff1a;Mark 1.注册标记 在项目根目录下创建固定名为 pytest.ini 的配置文件&#xff0c;文件格式需要加上 [pytest] &#xff0c;然后通过 markers 注册自定义标记 2.贴上标记 通过pytest加上装饰器&#xff0c;然后pytest.mark.XX配置自定义的标记&#xff0c;一个…

【C++】——多态

一.多态的概念 1.多态 多态(polymorphism)的概念&#xff1a;通俗的来说&#xff0c;就是多种形态。多态分为静态多态(编译时多态)和动态多态(运行时多态)&#xff0c;而我们讲的多态大部分都是动态多态。 静态多态主要就是我们前面了解过的函数模板和函数重载&#xff0c;它…

Linux基础4-进程4(环境变量,命令行参数详解)

上篇文章:Linux基础4-进程3(进程优先级&#xff0c;竞争&#xff0c;独立&#xff0c;并行&#xff0c;并发&#xff0c;进程切换)-CSDN博客 本章重点: Linux中环境变量的理解和使用 目录 一. 环境变量概念和查看环境变量 1.1 环境变量概念 1.2 查看环境变量 二. 获取环境变…

【复平面】-复数相乘的几何性质

文章目录 从数学上证明1. 计算乘积 z 1 ⋅ z 2 z_1 \cdot z_2 z1​⋅z2​2. 应用三角恒等式3. 得出结果 从几何角度证明1.给出待乘的复数 u i u_i ui​2.给出任意复数 l l l3.复数 l l l 在不同坐标轴下的表示图 首先说结论&#xff1a; 在复平面中&#xff0c;两个复数&a…

如何将现有VUE项目所有包更新到最新稳定版

更新有风险,Enter要谨慎!!! 要将项目中的所有 npm 包更新到最新稳定版&#xff0c;可以使用 npm-check-updates 工具。以下是具体步骤&#xff1a; 步骤一&#xff1a;安装 npm-check-updates 首先&#xff0c;全局安装 npm-check-updates 工具&#xff1a; npm install -g…

excel常用技能

1.基础技能 1.1 下拉框设置 a. 选中需要设置的列或单元格&#xff0c;数据 ---》 数据验证 b.验证条件 ---> 序列&#xff08;多个值逗号隔开&#xff09; 2.函数 2.1 统计函数-count a.count(区域&#xff0c;区域&#xff0c;......) 统计数量&#xff0c;只针…