当前位置: 首页 > news >正文

Swagger之Hello World !

目录

■1.前言・Swagger介绍

■2.例子,如果基于Spring Boot项目,实现Swagger---非常简单

2.1.已有的SpringBoot项目

2.2.修改POM文件

2.3.添加Config文件

2.4.访问

2.5.访问效果以及对应的代码

效果

代码

2.6.类的标注

2.7.启动Log

2.8.xxx

2.9.xxx

■3.非Spring 项目中,Swagger的运用

3.0.Sping介绍

3.1.简介

3.2.SwaggerConfig

3.3.配置Swagger相关Bean

3.4.访问

3.5.对于非Spring工程,如何加载Swagger的配置类

3.6.Web工程,如何配置,能让一个类在服务启动的时候,被加载

方法1(最基本方法):

方法2:@WebListener

3.7.ServletContextListener


====

■1.前言・Swagger介绍

项目去IF设计书化,使用Swagger生成API接口。

先了解一下Swagger

Swagger是一组开源工具,用于设计、构建、文档化和测试RESTful API。它提供了一种基于JSON或YAML格式的API描述语言,可以定义API的操作、参数、响应和安全规范,并生成易于阅读和交互的API文档。Swagger还提供了一组可视化工具,包括Swagger UI和Swagger Editor,可以帮助开发人员快速构建和测试API。使用Swagger可以提高API的可读性、可维护性和交互性,从而促进API的开发和使用。

学习资料

官方文档:Swagger Documentation

Swagger Editor:Swagger Editor

Swagger UI:REST API Documentation Tool | Swagger UI

使用Swagger自动生成API文档:https://www.cnblogs.com/kevingrace/p/7883099.html

使用Swagger构建RESTful API:https://www.jianshu.com/p/9b1b75b4f3ea

Swagger入门教程:https://www.cnblogs.com/panpanwelcome/p/10094733.html

Swagger的使用及其原理简介:https://www.jianshu.com/p/a5f6e5e9f4cf

Spring Boot集成Swagger:https://www.jianshu.com/p/03c7b314d1f8

===

■2.例子,如果基于Spring Boot项目,实现Swagger---非常简单

2.1.已有的SpringBoot项目

SpringBoot + Thymeleaf 之 HelloWorld_sun0322的博客-CSDN博客

====

2.2.修改POM文件

	 <!--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>

===

2.3.添加Config文件

package com.sxz.test.one.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;import com.google.common.base.Predicates;import springfox.documentation.builders.ApiInfoBuilder;
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 springfox.documentation.swagger2.annotations.EnableSwagger2;@Configuration
@EnableSwagger2
public class Swagger2Config {/*** 创建API应用* apiInfo() 增加API相关信息* 通过select()函数返回一个ApiSelectorBuilder实例,用来控制哪些接口暴露给Swagger来展现,* 指定扫描的包路径来定义指定要建立API的目录。* @return*/@Beanpublic Docket coreApiConfig(){return new Docket(DocumentationType.SWAGGER_2).apiInfo(adminApiInfo()).groupName("xxxApi").select()//只显示user下面的路径.paths(Predicates.and(PathSelectors.regex("/user/.*"))).apis(RequestHandlerSelectors.basePackage("com.sxz")).build();}private ApiInfo adminApiInfo(){return new ApiInfoBuilder().title("XXXX--api文档").description("My Spring boot Api 介绍。。。。。").version("1.0").contact(new Contact("zhangsan","https://blog.csdn.net/sxzlc","123456789@sun.com")).build();}
}

===

2.4.访问

https://10.10.10.194/swagger-ui.html#/

===

2.5.访问效果以及对应的代码

效果

代码

package com.sxz.test.one.controller;import java.util.List;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;import com.sxz.test.one.entity.User;
import com.sxz.test.one.service.UserService;import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import io.swagger.annotations.ApiParam;
import lombok.extern.slf4j.Slf4j;@Api(value = "User Info")
@Controller
@RequestMapping("/user")
@Slf4j
public class UserController2 {@AutowiredUserService userService;@ApiOperation(value = "All User Info",notes = "Find All User Info")@RequestMapping("/findAll2")public String findAll(@ApiParam(value = "Model",required = true,example = "example model")Model model){log.info("Hello World !-------!");List<User> userList = userService.findAll();model.addAttribute("userList",userList);return "helloThymeleafMyBatis.html";}
}

xxx

2.6.类的标注

xxxx

在完成了上述配置后,其实已经可以生产文档内容,但是这样的文档主要针对请求本身,描述的主要来源是函数的命名,通常需要自己增加一些说明来丰富文档内容。

Swagger使用的注解及其说明:

@Api:用在类上,说明该类的作用。
@ApiOperation:注解来给API增加方法说明。
@ApiParam:定义在参数上
@ApiResponses:用于表示一组响应
@ApiResponse:用在@ApiResponses中,一般用于表达一个错误的响应信息
l code:数字,例如400
l message:信息,例如"请求参数没填好"
l response:抛出异常的类

@ApiModel:描述一个Model的信息(一般用在请求参数无法使用@ApiImplicitParam注解进行描述的时候)

l @ApiModelProperty:描述一个model的属性

@ApiImplicitParams: 用在方法上包含一组参数说明。

@ApiImplicitParam:用来注解来给方法入参增加说明。

@ApiImplicitParam的参数说明:

paramType:指定参数放在哪个地方

header:请求参数放置于Request Header,使用@RequestHeader获取 query:请求参数放置于请求地址,使用@RequestParam获取 path:(用于restful接口)–>请求参数的获取:@PathVariable body:(不常用) form(不常用)

name:参数名

dataType:参数类型

required:参数是否必须传

true | false

value:说明参数的意思

defaultValue:参数的默认值

xxxx

2.7.启动Log

。。。。。。。

    [2023-07-31 19:04:20.990] [level: INFO] [Thread: main] [ Class:org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext >> Method: prepareWebApplicationContext:285 ]
    INFO:Root WebApplicationContext: initialization completed in 5824 ms

    [2023-07-31 19:04:24.578] [level: INFO] [Thread: main] [ Class:springfox.documentation.spring.web.PropertySourcedRequestMappingHandlerMapping >> Method: initHandlerMethods:69 ]
    INFO:Mapped URL path [/v2/api-docs] onto method [springfox.documentation.
swagger2.web.Swagger2Controller#getDocumentation(String, HttpServletRequest)]

    [2023-07-31 19:04:24.889] [level: INFO] [Thread: main] [ Class:org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor >> Method: initialize:181 ]
    INFO:Initializing ExecutorService 'applicationTaskExecutor'

    [2023-07-31 19:04:25.143] [level: INFO] [Thread: main] [ Class:org.springframework.boot.autoconfigure.web.servlet.WelcomePageHandlerMapping >> Method: <init>:53 ]
    INFO:Adding welcome page: class path resource [static/index.html]

。。。。。。。

2.8.xxx

xxxx

2.9.xxx

xxxx

===

■3.非Spring 项目中,Swagger的运用

3.0.Sping介绍

Spring Boot,Sprint Batch,ThymeLeaf 学习_sun0322的博客-CSDN博客

xxxx

3.1.简介

Swagger是一种用于创建、文档化和测试Restful API的工具,不限于特定的框架或工程。即使你的web工程不是基于Spring或Spring Boot的,你仍然可以使用Swagger来实现API文档化和测试。

如果你的web工程不是基于Spring或Spring Boot的,你可以使用Swagger2配置Swagger,提供API文档化和测试的功能。下面给出一个示例配置:

xxxx

3.2.SwaggerConfig

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
@EnableSwagger2
public class SwaggerConfig {public Docket api() {return new Docket(DocumentationType.SWAGGER_2).select().apis(RequestHandlerSelectors.basePackage("com.example.api")) // 修改为你的API接口所在的包路径.paths(PathSelectors.any()).build().apiInfo(apiInfo());}private ApiInfo apiInfo() {return new ApiInfoBuilder().title("API文档").description("描述你的API").version("1.0").build();}
}

xxxx

3.3.配置Swagger相关Bean

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class AppConfig {@Beanpublic SwaggerConfig swaggerConfig() {return new SwaggerConfig();}
}

===

3.4.访问

启动工程并访问Swagger UI:当你的工程启动后,你可以通过访问Swagger UI来查看生成的API文档。Swagger UI的访问地址一般是http://localhost:port/swagger-ui.html,其中port是你的web应用的端口号。

==

需要注意的是,以上示例代码是以Java配置的方式示范,如果你的工程使用其他配置方式(如XML配置),你需要相应地进行调整。

==

以上是一个基本的Swagger2配置示例,通过在工程中配置Swagger,你可以实现API文档化和测试的功能。具体的配置细节可能会因为工程的不同而有所差异,你可以根据Swagger官方文档或相关的API文档进行调整和修改。

xxxx

3.5.对于非Spring工程,如何加载Swagger的配置类

对于非Spring工程,如果你想要让程序加载Swagger的配置类

public class MainClass {public static void main(String[] args) throws Exception {Class<?> swaggerConfigClass = Class.forName("你的包名.SwaggerConfig"); // 修改为SwaggerConfig类所在的包名}
}

xxx

调用SwaggerConfig类的api()方法:通过反射调用SwaggerConfig类中的api()方法来获取Docket对象。

public class MainClass {public static void main(String[] args) throws Exception {Class<?> swaggerConfigClass = Class.forName("你的包名.SwaggerConfig"); // 修改为SwaggerConfig类所在的包名Object swaggerConfigObject = swaggerConfigClass.newInstance();Method apiMethod = swaggerConfigClass.getDeclaredMethod("api");Docket docket = (Docket) apiMethod.invoke(swaggerConfigObject);}
}

使用Docket对象进行Swagger的配置和初始化。

public class MainClass {public static void main(String[] args) throws Exception {Class<?> swaggerConfigClass = Class.forName("你的包名.SwaggerConfig"); // 修改为SwaggerConfig类所在的包名Object swaggerConfigObject = swaggerConfigClass.newInstance();Method apiMethod = swaggerConfigClass.getDeclaredMethod("api");Docket docket = (Docket) apiMethod.invoke(swaggerConfigObject);// 根据你的实际需求,进行其他的Swagger配置(例如设置路径,设置API文档信息等)// 初始化SwaggerSwagger swagger = docket.initialize();// 可以根据需要将Swagger对象保存下来,供其他需要使用Swagger的地方使用}
}

sss

以上是在非Spring工程中手动加载Swagger配置类的方法,你可以根据实际情况进行调整。需要注意的是,这种方式需要手动处理加载和初始化,相对于Spring工程中的自动注入来说,比较繁琐。如果你的项目是非常简单的项目,也可以考虑使用Spring或Spring Boot来简化Swagger的配置和集成。

3.6.Web工程,如何配置,能让一个类在服务启动的时候,被加载

方法1(最基本方法):

首先,创建一个类实现javax.servlet.ServletContextListener接口并重写contextInitialized方法,该方法会在Web应用启动时被调用。在该方法中,你可以执行你想要在服务启动时加载的逻辑。

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;public class MyServletContextListener implements ServletContextListener {@Overridepublic void contextInitialized(ServletContextEvent servletContextEvent) {// 这里写你想要在服务启动时加载的逻辑}@Overridepublic void contextDestroyed(ServletContextEvent servletContextEvent) {// 服务关闭时的逻辑}
}

然后,在web.xml中配置该监听器。找到web.xml文件,添加以下代码:

<listener><listener-class>你的包名.MyServletContextListener</listener-class> <!-- 修改为你的类的全限定名 -->
</listener>

xxx

当你的Web应用启动时,MyServletContextListener类的contextInitialized方法将会被调用,在该方法中可以执行你想要在服务启动时加载的逻辑。

===

方法2:@WebListener

注意:在最新的Servlet规范和Servlet 3.0之后,你还可以使用注解的方式配置ServletContextListener你可以在想要加载的类上添加@WebListener注解,并将其放置在类上方。

xxx

import javax.servlet.annotation.WebListener;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;@WebListener
public class MyServletContextListener implements ServletContextListener {@Overridepublic void contextInitialized(ServletContextEvent servletContextEvent) {// 这里写你想要在服务启动时加载的逻辑}@Overridepublic void contextDestroyed(ServletContextEvent servletContextEvent) {// 服务关闭时的逻辑}
}

这种方式更简洁,并且不需要在web.xml中进行配置。

无论是使用web.xml配置还是使用注解配置ServletContextListener,都可以达到在Web服务启动时加载某个类的目的。

xxx

3.7.ServletContextListener

 ===

 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.package javax.servlet;import java.util.EventListener;/** * Interface for receiving notification events about ServletContext* lifecycle changes.** <p>In order to receive these notification events, the implementation* class must be either declared in the deployment descriptor of the web* application, annotated with {@link javax.servlet.annotation.WebListener},* or registered via one of the addListener methods defined on* {@link ServletContext}.** <p>Implementations of this interface are invoked at their* {@link #contextInitialized} method in the order in which they have been* declared, and at their {@link #contextDestroyed} method in reverse* order.** @see ServletContextEvent** @since Servlet 2.3*/
public interface ServletContextListener extends EventListener {/*** Receives notification that the web application initialization* process is starting.** <p>All ServletContextListeners are notified of context* initialization before any filters or servlets in the web* application are initialized.** @param sce the ServletContextEvent containing the ServletContext* that is being initialized*/public void contextInitialized(ServletContextEvent sce);/*** Receives notification that the ServletContext is about to be* shut down.** <p>All servlets and filters will have been destroyed before any* ServletContextListeners are notified of context* destruction.** @param sce the ServletContextEvent containing the ServletContext* that is being destroyed*/public void contextDestroyed(ServletContextEvent sce);
}

 xxx

===

相关文章:

Swagger之Hello World !

目录 ■1&#xff0e;前言・Swagger介绍 ■2&#xff0e;例子&#xff0c;如果基于Spring Boot项目&#xff0c;实现Swagger---非常简单 2&#xff0e;1&#xff0e;已有的SpringBoot项目 2&#xff0e;2&#xff0e;修改POM文件 2&#xff0e;3&#xff0e;添加Config文件…...

VSCode SSH远程连接与删除

1.ubuntu设置 安装SSH服务并获取远程访问的IP地址 在Ubuntu系统中&#xff0c;“CtrlAltT”打开终端工具&#xff0c;执行如下命令安装SSH服务。 sudo apt-get install openssh-server如果安装失败则先安装依赖项。 2.VS Code 设置 2.1安装与设置Remote SSH 打开Windows系…...

面试典中典之线程池的七大参数

文章目录 一、七大元素解释1.corePoolSize&#xff08;核心线程数&#xff09;&#xff1a;2.maximumPoolSize&#xff08;最大线程数&#xff09;&#xff1a;3.keepAliveTime&#xff08;线程空闲时间&#xff09;&#xff1a;4.unit&#xff08;时间单位&#xff09;&#x…...

Maven如何创建Java web项目(纯干货版)!!!

1.创建Maven项目。 2.创建完成后会来到这个界面。 3.在src/main目录下&#xff0c;建立webapp / WEB-INF/web.xml文件&#xff0c;并在web.xml文件中写入以下内容&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <web-app xmlns"http…...

linux进程间通信的方式及特点

1&#xff1a;信号 描述&#xff1a; 信号是linux上的一种软中断通信机制&#xff0c;可以向指定进程发送通知&#xff0c;总共有64种信号&#xff1b; 特点&#xff1a; 信号只能作为通知使用&#xff0c;没办法传递数据&#xff1b; 2&#xff1a;socket套接字 描述&#…...

京东的成功秘诀:找到自己独特而有效的商业模式

你知道京东为什么能够从一个卖电器的小网站发展成为中国最大的电商平台吗&#xff1f;如果京东一开始靠卖电器赚钱&#xff0c;不可能有今天&#xff0c;在十几年刘强东刚创业的时候&#xff0c;如果京东靠卖电器赚钱&#xff0c;你知道想当年的国美黄光裕和想当年的苏宁的张近…...

全局ip代理安全吗? 手机设置全局代理方法详解

全局IP代理并不一定是安全的&#xff0c;因为全局IP代理会将所有网络流量都通过代理服务器进行转发&#xff0c;包括敏感信息和隐私数据。如果代理服务器受到黑客攻击或存在安全漏洞&#xff0c;可能会导致数据泄露和其他安全问题。因此&#xff0c;在使用全局IP代理时&#xf…...

Clion开发Stm32之温湿度传感器(DHT11)驱动编写

前言 涵盖之前文章: Clion开发STM32之HAL库GPIO宏定义封装(最新版)Clion开发stm32之微妙延迟(采用nop指令实现)Clion开发STM32之日志模块(参考RT-Thread) DHT11驱动文件 头文件 /*******************************************************************************Copyrig…...

位操作相关的函数(C++)

目录 popcount函数 bitset类模板 __builtin_popcount函数 popcount函数 在C中&#xff0c;std::popcount函数是用来计算一个整数二进制表示中包含的1的个数。不过要注意&#xff0c;这个函数是C20标准引入的&#xff0c;因此在使用之前&#xff0c;要先确保编译器支持C20标…...

arm 函数栈回溯

大概意思就是arm每个函数开始都会将PC、LR、SP以及FP四个寄存器入栈。 下面我们看一下这四个寄存器里面保存的是什么内存 arm-linux-gnueabi-gcc unwind.c -mapcs -w -g -o unwind&#xff08;需要加上-mapcs才会严格按照上面说的入栈&#xff09; #include <stdio.h> …...

30个前端开发中常用的JavaScript函数

&#x1f9d1;‍&#x1f4bb;作者名称&#xff1a;DaenCode &#x1f3a4;作者简介&#xff1a;啥技术都喜欢捣鼓捣鼓&#xff0c;喜欢分享技术、经验、生活。 &#x1f60e;人生感悟&#xff1a;尝尽人生百味&#xff0c;方知世间冷暖。 前言 在前端开发中通常会用到校验函数…...

基于量子同态加密的改进多方量子私有比较

摘要量子同态加密在隐私保护方面具有明显的优势。本文提出了一种改进的基于量子同态加密的多方量子私钥比较协议。首先&#xff0c;引入可信密钥中心&#xff0c;安全辅助加密密钥的分发和解密密钥的更新&#xff0c;同时防止恶意服务器发布虚假结果的攻击;在保证所有参与者得到…...

解决mysqld服务启动失败

原因如下&#xff1a; 1、进程占用 首先查看下mysql进程: ps -aux | grep mysql有进程号占用了&#xff0c;kill 这个进程号 再重启服务 2、所有者和所属组为mysql 查看/usr/local/MySQL/data/mysqld.pid所有者和所属组是否为mysql 原来是权限有问题&#xff0c…...

【前端知识】React 基础巩固(四十)——Navigate导航

React 基础巩固(四十)——Navigate导航 一、Navigate的基本使用 新建Login页面&#xff0c;在Login中引入Navigate&#xff0c;实现点击登陆按钮跳转至/home路径下&#xff1a; import React, { PureComponent } from "react"; import { Navigate } from "reac…...

文件IO练习

一、用read函数完成文件大小计算 #include <stdio.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> int main(int argc, const char *argv[]) {int fd open("./1.tx…...

初识FreeRTOS入门,对FreeRTOS简介、任务调度、内存管理、通信机制以及IO操作,控制两个led不同频率闪烁

当代嵌入式系统的开发越来越复杂&#xff0c;实时性要求也越来越高。为了满足这些需求&#xff0c;开发者需要使用实时操作系统&#xff08;RTOS&#xff09;&#xff0c;其中一个流行的选择是FreeRTOS&#xff08;Free Real-Time Operating System&#xff09;。本篇博客将详细…...

STM32CUBUMX配置FLASH(W25Q128)--保姆级教程

———————————————————————————————————— ⏩ 大家好哇&#xff01;我是小光&#xff0c;嵌入式爱好者&#xff0c;一个想要成为系统架构师的大三学生。 ⏩最近在开发一个STM32H723ZGT6的板子&#xff0c;使用STM32CUBEMX做了很多驱动&#x…...

【Golang 接口自动化04】 解析接口返回JSON串

目录 前言 解析到结构体 json数据与struct字段是如何相匹配的呢&#xff1f; 解析到interface Go类型和JSON类型 实例代码 simpleJson 总结 资料获取方法 前言 上一次我们一起学习了如何解析接口返回的XML数据&#xff0c;这一次我们一起来学习JSON的解析方法。 JSO…...

EPPlus与Microsoft.Office.Interop.Excel的使用区别

文章目录 代码的使用区别EPPlus的工作原理Microsoft.Office.Interop.Excel的使用原理代码的使用区别 static void ExportToExcel(List<(string, double, double)> list, string outputFilePath){//Microsoft.Office.Interop.Excel的使用 /* Excel.Application excelAp…...

ncrack工具使用说明

介绍 网络认证破解工具。 Ncrack是用于网络身份验证破解的开源工具。 它设计为使用可适应不同网络情况的动态引擎进行高速并行破解。 Ncrack还可以针对特殊情况进行广泛的微调,尽管默认参数的通用性足以覆盖几乎所有情况。 它建立在模块化架构上,可以轻松扩展以支持其他协议…...

web vue 项目 Docker化部署

Web 项目 Docker 化部署详细教程 目录 Web 项目 Docker 化部署概述Dockerfile 详解 构建阶段生产阶段 构建和运行 Docker 镜像 1. Web 项目 Docker 化部署概述 Docker 化部署的主要步骤分为以下几个阶段&#xff1a; 构建阶段&#xff08;Build Stage&#xff09;&#xff1a…...

java_网络服务相关_gateway_nacos_feign区别联系

1. spring-cloud-starter-gateway 作用&#xff1a;作为微服务架构的网关&#xff0c;统一入口&#xff0c;处理所有外部请求。 核心能力&#xff1a; 路由转发&#xff08;基于路径、服务名等&#xff09;过滤器&#xff08;鉴权、限流、日志、Header 处理&#xff09;支持负…...

智能在线客服平台:数字化时代企业连接用户的 AI 中枢

随着互联网技术的飞速发展&#xff0c;消费者期望能够随时随地与企业进行交流。在线客服平台作为连接企业与客户的重要桥梁&#xff0c;不仅优化了客户体验&#xff0c;还提升了企业的服务效率和市场竞争力。本文将探讨在线客服平台的重要性、技术进展、实际应用&#xff0c;并…...

对WWDC 2025 Keynote 内容的预测

借助我们以往对苹果公司发展路径的深入研究经验&#xff0c;以及大语言模型的分析能力&#xff0c;我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际&#xff0c;我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测&#xff0c;聊作存档。等到明…...

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放

简介 前面两期文章我们介绍了I2S的读取和写入&#xff0c;一个是通过INMP441麦克风模块采集音频&#xff0c;一个是通过PCM5102A模块播放音频&#xff0c;那如果我们将两者结合起来&#xff0c;将麦克风采集到的音频通过PCM5102A播放&#xff0c;是不是就可以做一个扩音器了呢…...

全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比

目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec&#xff1f; IPsec VPN 5.1 IPsec传输模式&#xff08;Transport Mode&#xff09; 5.2 IPsec隧道模式&#xff08;Tunne…...

在QWebEngineView上实现鼠标、触摸等事件捕获的解决方案

这个问题我看其他博主也写了&#xff0c;要么要会员、要么写的乱七八糟。这里我整理一下&#xff0c;把问题说清楚并且给出代码&#xff0c;拿去用就行&#xff0c;照着葫芦画瓢。 问题 在继承QWebEngineView后&#xff0c;重写mousePressEvent或event函数无法捕获鼠标按下事…...

return this;返回的是谁

一个审批系统的示例来演示责任链模式的实现。假设公司需要处理不同金额的采购申请&#xff0c;不同级别的经理有不同的审批权限&#xff1a; // 抽象处理者&#xff1a;审批者 abstract class Approver {protected Approver successor; // 下一个处理者// 设置下一个处理者pub…...

Webpack性能优化:构建速度与体积优化策略

一、构建速度优化 1、​​升级Webpack和Node.js​​ ​​优化效果​​&#xff1a;Webpack 4比Webpack 3构建时间降低60%-98%。​​原因​​&#xff1a; V8引擎优化&#xff08;for of替代forEach、Map/Set替代Object&#xff09;。默认使用更快的md4哈希算法。AST直接从Loa…...

Redis:现代应用开发的高效内存数据存储利器

一、Redis的起源与发展 Redis最初由意大利程序员Salvatore Sanfilippo在2009年开发&#xff0c;其初衷是为了满足他自己的一个项目需求&#xff0c;即需要一个高性能的键值存储系统来解决传统数据库在高并发场景下的性能瓶颈。随着项目的开源&#xff0c;Redis凭借其简单易用、…...