Spring Boot Web技术栈(官网文档解读)
摘要
Spring Boot框架既支持传统的Servlet技术栈,也支持新兴的响应式(Reactive)技术栈。本篇文章将详细讲述Spring Boot 对两种技术栈的详细支持和使用。
Servlet
概述
基于Java Servlet API构建,它依赖于传统的阻塞I/O模型,每个HTTP请求都会分配一个线程来处理,直到响应返回给客户端。这种模型适用于大多数传统的Web应用,其中大部分操作是同步的,并且涉及到数据库查询或文件读写等I/O密集型任务。
-
特点
-
同步编程模型:代码按照顺序执行,等待每个操作完成后再继续下一个操作。
-
线程池机制:使用线程池管理请求处理线程,但当并发用户数增加时,可能会导致资源耗尽问题。
-
成熟的生态系统:拥有丰富的库和框架支持(如Spring MVC),并且已经被广泛应用于企业级应用中。
-
-
适用场景:适合那些对实时性和高并发要求不是特别高的应用场景,尤其是当业务逻辑较为复杂或者需要大量的数据处理时。
Spring Boot 对Servlet技术栈的实现架构
Spring MVC
简介
Spring MVC 是 Spring Framework 的核心组件之一,专为构建基于 Servlet 技术的 Web 应用程序而设计。Spring Boot 整合了 Spring MVC 框架,以此提供对传统 Web Servlet 技术栈的全面支持,使得开发者能够便捷地构建和运行基于 Servlet 的 Web 应用。
实现方式
添加依赖Spring-boot-starter-web。Spring Boot 默认使用Servlet 技术栈的Spring MVC 架构。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId>
</dependency>
官网提供了两种实现Spring MVC 架构的示例。第一种是常见的采用 @Controller 或 @RestController 注解的实现方式。
package person.wend.springbootlearnexample.controller;import org.springframework.web.bind.annotation.*;@RestController
@RequestMapping("/users")
public class MyRestController {@GetMapping("/{userId}")public void getUser(@PathVariable Long userId) {System.out.println("userId = " + userId);}@GetMapping("/{userId}/customers")public void getUserCustomers(@PathVariable Long userId) {System.out.println("userId = " + userId);}@DeleteMapping("/{userId}")public void deleteUser(@PathVariable Long userId) {System.out.println("userId = " + userId);}}
第二种是基于函数式编程的初始化配置路由的方式。
package person.wend.springbootlearnexample.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.MediaType;
import org.springframework.web.servlet.function.RequestPredicate;
import org.springframework.web.servlet.function.RouterFunction;
import org.springframework.web.servlet.function.ServerResponse;import static org.springframework.web.servlet.function.RequestPredicates.accept;
import static org.springframework.web.servlet.function.RouterFunctions.route;@Configuration(proxyBeanMethods = false)
public class MyRountingConfig {private static final RequestPredicate ACCEPT_JSON = accept(MediaType.APPLICATION_JSON);@Beanpublic RouterFunction<ServerResponse> routerFunction(MyUserHandler userHandler) {return route().GET("/{user}", ACCEPT_JSON, userHandler::getUser).GET("/{user}/customers", ACCEPT_JSON, userHandler::getUserCustomers).DELETE("/{user}", ACCEPT_JSON, userHandler::deleteUser).build();}
}
package person.wend.springbootlearnexample.config;import org.springframework.stereotype.Component;
import org.springframework.web.servlet.function.ServerRequest;
import org.springframework.web.servlet.function.ServerResponse;@Component
public class MyUserHandler {public ServerResponse getUser(ServerRequest request) {System.out.println("request = " + request);return ServerResponse.ok().body("getUser");}public ServerResponse getUserCustomers(ServerRequest request) {System.out.println("request = " + request);return ServerResponse.ok().body("getUserCustomers");}public ServerResponse deleteUser(ServerRequest request) {System.out.println("request = " + request);return ServerResponse.ok().body("deleteUser");}}
自动装配Spring MVC
Spring Boot 为 Spring MVC 提供了自动配置功能,该功能适用于大多数应用程序。它取代了对 @EnableWebMvc 的需求,并且两者不能同时使用。除了 Spring MVC 的默认配置之外,自动配置还提供了以下特性:
-
视图解析器相关:包含了 ContentNegotiatingViewResolver 和 BeanNameViewResolver 这两个 Bean。ContentNegotiatingViewResolver 可以根据请求的媒体类型(如 Accept 头信息)和可提供的视图,来决定使用哪个视图进行响应。而 BeanNameViewResolver 会根据视图名称来查找相应的 Bean 作为视图,例如,如果视图名称是 “myView”,它会查找名为 “myView” 的 Bean 作为视图进行渲染。
-
静态资源服务:支持提供静态资源服务,其中包括对 WebJars 的支持。WebJars 是一种将前端库(如 jQuery、Bootstrap 等)打包成 JAR 文件的方式,这样可以方便地将它们作为依赖添加到 Java 项目中,并且 Spring Boot 会自动将其作为静态资源进行处理,使得前端资源的管理更加方便。
-
转换器、通用转换器和格式化器的自动注册:会自动注册 Converter、GenericConverter 和 Formatter 这些 Bean。Converter 可用于将一种数据类型转换为另一种数据类型,例如将字符串转换为日期类型;GenericConverter 是一种更通用的转换机制,提供了更灵活的转换方式;Formatter 则侧重于对数据进行格式化操作,比如将日期对象格式化为特定的字符串形式,方便在视图中显示,它们的自动注册可以简化开发过程,减少手动配置的工作量。
-
HttpMessageConverters 的支持:支持 HttpMessageConverters。HttpMessageConverters 用于将 HTTP 请求或响应中的数据在 Java 对象和 HTTP 消息体之间进行转换。例如,将 Java 对象转换为 JSON 或 XML 格式并发送到客户端,或者将客户端发送的 JSON 或 XML 数据转换为 Java 对象,这对于处理 RESTful API 非常重要。
-
MessageCodesResolver 的自动注册:自动注册 MessageCodesResolver。MessageCodesResolver 通常用于生成和解析消息代码,在进行数据校验等场景中,它可以根据错误信息生成特定的消息代码,以便更好地处理错误信息,使应用程序可以根据消息代码进行不同的错误处理操作。
-
静态 index.html 支持:支持静态的 index.html 文件。这意味着如果在静态资源目录下存在 index.html 文件,当用户访问应用程序的根路径时,Spring Boot 会直接提供该文件作为响应,方便进行静态页面的展示,这对于一些单页应用程序(SPA)或者静态站点的部署非常有用。
-
自动使用 ConfigurableWebBindingInitializer Bean:自动使用 ConfigurableWebBindingInitializer Bean(本文后续会提及)。这个 Bean 可以对 Web 请求参数绑定进行初始化操作,例如可以设置数据绑定的日期格式、绑定过程中忽略的字段等,确保请求参数可以正确地绑定到 Java 对象上。
如果您想要保留 Spring Boot 的这些 MVC 自定义特性,并进行更多的 MVC 自定义操作(如拦截器、格式化器、视图控制器以及其他功能),您可以添加自己的 @Configuration 类,其类型为 WebMvcConfigurer,但不要使用 @EnableWebMvc。
如果您想要提供 RequestMappingHandlerMapping、RequestMappingHandlerAdapter 或 ExceptionHandlerExceptionResolver 的自定义实例,同时又想保留 Spring Boot 的 MVC 自定义特性,可以声明一个类型为 WebMvcRegistrations 的 Bean,并使用它来提供这些组件的自定义实例。这些自定义实例将接受 Spring MVC 的进一步初始化和配置。如果您想参与并在需要时覆盖后续的处理过程,可以使用 WebMvcConfigurer。
如果您不想使用自动配置,并且想要完全掌控 Spring MVC,可以添加自己的、带有 @EnableWebMvc 注解的 @Configuration 类。或者,按照 @EnableWebMvc 的 API 文档中描述的那样,添加自己的带有 @Configuration 注解的 DelegatingWebMvcConfiguration。
Jersey
Jersey 是一个开源的、用于构建 RESTful Web 服务的框架,它是 Java API for RESTful Web Services(JAX-RS)的参考实现。它提供了一组强大的工具和 API,使开发人员能够轻松地创建、部署和管理 RESTful 服务。
Spring Boot 集成 Jersey
添加依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jersey</artifactId>
</dependency>
注册端点
package person.wend.springbootlearnexample.config;import org.glassfish.jersey.server.ResourceConfig;
import org.springframework.stereotype.Component;
import person.wend.springbootlearnexample.controller.JerseyController;
import person.wend.springbootlearnexample.controller.UserJerseyController;@Component
public class MyJerseyConfig extends ResourceConfig {public MyJerseyConfig() {register(JerseyController.class);register(UserJerseyController.class);}
}
Jersey 基于 JAX-RS 标准
Jersey 遵循 JAX-RS 规范,这意味着它提供了标准的注解和 API 来开发 RESTful Web 服务。例如,使用 @Path
注解来定义资源的 URI 路径,@GET
、@POST
、@PUT
、@DELETE
等注解来定义 HTTP 方法。以下是一个简单的示例:
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;@Path("/hello")
public class HelloResource {@GET@Produces(MediaType.TEXT_PLAIN)public String sayHello() {return "Hello, World!";}
}
在上述代码中:
-
@Path("/hello")
定义了资源的 URI 路径为/hello
。 -
@GET
表示这个方法将处理 HTTP 的 GET 请求。 -
@Produces(MediaType.TEXT_PLAIN)
表示该方法将产生纯文本类型的响应。
强大的请求和响应处理
Jersey 支持多种媒体类型的请求和响应处理,包括 JSON、XML、HTML、纯文本等。通过 @Consumes
和 @Produces
注解可以指定请求和响应的数据类型。例如:
import javax.ws.rs.Consumes;
import javax.ws.rs.GET;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;@Path("/user")
public class UserResource {@GET@Produces(MediaType.APPLICATION_JSON)public User getUser() {User user = new User("John", 30);return user;}@POST@Consumes(MediaType.APPLICATION_JSON)@Produces(MediaType.APPLICATION_JSON)public Response createUser(User user) {// 处理用户创建逻辑return Response.status(Response.Status.CREATED).entity(user).build();}
}
-
@Consumes(MediaType.APPLICATION_JSON)
表示createUser
方法接受 JSON 类型的请求体。 -
@Produces(MediaType.APPLICATION_JSON)
表示该方法将返回 JSON 类型的响应。
Spring Boot 对Servlet 技术栈的嵌入式容器支持
当您希望采用传统的Servlet栈来构建应用时,您需要添加spring-boot-starter-web
依赖,该依赖默认集成了spring-boot-starter-tomcat
,意味着您的应用将使用Tomcat作为Web容器。此外,Spring Boot还支持使用Jetty和Undertow作为可选的Web容器。
Spring Boot 替换容器示例
以下是一个使用jetty 替换默认的tomcat 的Maven 依赖配置示例。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><!-- Exclude the Tomcat dependency --><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-tomcat</artifactId></exclusion></exclusions>
</dependency>
<!-- Use Jetty instead -->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jetty</artifactId>
</dependency>
Reactive
概述
Reactive 引入了非阻塞I/O模型和事件驱动架构,旨在提高系统的可伸缩性和响应能力。它允许应用程序以异步方式处理请求,从而可以在更少的线程上处理更多的并发连接。这使得Reactive Stack非常适合构建微服务架构下的高性能Web服务。
-
特点
-
异步编程模型:采用回调函数或观察者模式实现异步操作,避免了长时间等待资源释放的问题。
-
背压机制:通过Reactive Streams规范提供流控能力,确保上游不会过快地发送数据给下游,造成系统过载。
-
轻量级线程:利用协程(Coroutines)或其他轻量级线程替代传统线程,降低了上下文切换开销。
-
现代Web容器:通常运行在像Netty这样的非阻塞服务器上,也可以兼容Servlet 3.1+容器。
-
-
适用场景:特别适用于需要处理大量并发连接、低延迟要求的应用程序,例如实时聊天室、物联网平台、在线游戏等。
实现方式
-
添加依赖:我们需要引入 spring-boot-starter-webflux依赖以支持响应式Web 编程。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-webflux</artifactId>
</dependency>
-
依赖冲突问题 :需要注意的是,当我们在应用程序中同时添加了
spring-boot-starter-web
和spring-boot-starter-webflux
模块的依赖时, Spring Boot 会选择自动配置 Spring MVC,而不是 WebFlux。 -
如果项目中同时存在两个依赖,但是我们仍然想使用Webflux 架构,我们可以将所选应用程序类型设置为 SpringApplication.setWebApplicationType(WebApplicationType.REACTIVE) 来强制使用webflux 模块。
@SpringBootApplication
public class SpringBootLearnExampleApplication {public static void main(String[] args) {SpringApplication springApplication = new SpringApplication(SpringBootLearnExampleApplication.class);// 将应用设置为响应式应用springApplication.setWebApplicationType(WebApplicationType.REACTIVE);SpringApplication.run(SpringBootLearnExampleApplication.class, args);}}
-
Spring webFlux 的代码实现有两种方式,基于注释的和函数式编程,两种方式的实现代码与Spring MVC 架构的实现代码基本一致,只不过内部实现原理不同,WebFlux 的函数式是基于WebFlux.fn,而Spring MVC的函数式是基于WebMvc.fn。代码示例请参考上述Spring MVC 的代码实现示例。
Spring WebFlux 自动装配
Spring WebFlux的自动装配与 Spring MVC 的自动装配过程和功能均保持一致,需要注意的是关于WebFlux 自定义注解与MVC 框架不一致。比如如果您希望保留 Spring Boot 对 WebFlux 的这些自定义设置,同时进行更多的 WebFlux 自定义配置(如添加拦截器、自定义格式化器、设置视图控制器等其他功能),可以添加自定义的 @Configuration
类,使其实现 WebFluxConfigurer
接口,但不要使用 @EnableWebFlux
注解。
Spring Boot 对Reactive 技术栈的嵌入式容器支持
WebFlux默认采用Netty作为其网络通信框架。不过,您也可以选择配置Tomcat、Jetty或Undertow来替代Netty作为WebFlux的底层传输层。
Servlet Stack 与 Reactive Stack 比较
特性 | Servlet Stack | Reactive Stack |
---|---|---|
编程模型 | 同步 | 异步 |
线程管理 | 线程池 | 轻量级线程/协程 |
I/O模型 | 阻塞I/O | 非阻塞I/O |
性能优化 | 适用于中等规模并发 | 适用于大规模并发 |
生态系统 | 成熟稳定 | 相对年轻 |
Web容器 | Servlet容器(如Tomcat, Jetty) | 非阻塞服务器(如Netty) |
数据流处理 | 可以实现但较为复杂 | 内置支持 |
示例框架 | Spring MVC,Jersey | Spring WebFlux |
优雅关机
默认情况下,所有四个嵌入式 Web 服务器(Jetty、Reactor Netty、Tomcat 和 Undertow)以及反应式和基于 servlet 的 Web 应用程序均启用了优雅关闭。优雅关闭是关闭应用程序上下文的一部分,在停止SmartLifecyclebean 的最早阶段执行。此停止处理使用超时,提供宽限期,在此期间允许完成现有请求,但不允许新请求。
要配置超时时间,请配置spring.lifecycle.timeout-per-shutdown-phase
属性,如以下示例所示:
spring.lifecycle.timeout-per-shutdown-phase=20s
在宽限期内拒绝请求
不允许新请求的具体方式取决于所使用的 Web 服务器。实现可能会在网络层停止接受请求,或者它们可能会返回带有特定 HTTP 状态代码或 HTTP 标头的响应。使用持久连接也可以改变停止接受请求的方式。Jetty、Reactor Netty 和 Tomcat 将停止在网络层接受新请求。Undertow 将接受新连接,但会立即响应服务不可用 (503)。以下是四种Web 服务器的优雅关闭API 接口详情:
TomcatWebServer (Spring Boot 3.4.1 API)
NettyWebServer (Spring Boot 3.4.1 API)
JettyWebServer (Spring Boot 3.4.1 API)
UndertowWebServer (Spring Boot 3.4.1 API)
禁用优雅关机
server.shutdown=immediate
参考文献
Servlet Web Applications :: Spring Boot
相关文章:
Spring Boot Web技术栈(官网文档解读)
摘要 Spring Boot框架既支持传统的Servlet技术栈,也支持新兴的响应式(Reactive)技术栈。本篇文章将详细讲述Spring Boot 对两种技术栈的详细支持和使用。 Servlet 概述 基于Java Servlet API构建,它依赖于传统的阻塞I/O模型&…...
闲谭SpringBoot--ShardingSphere分布式事务探究
文章目录 0. 背景1. 未分库分表时2. 仅分表时3. 分库分表时3.1 不涉及分库表3.2 涉及分库表,且分库表处于一个库3.3 涉及分库表,且分库表处于多个库3.4 涉及分库表,且运行中某库停机 4. 小结 0. 背景 接上篇文章《闲谭SpringBoot–ShardingS…...
计算机网络之---TCP报文段
TCP报文段 TCP报文段是TCP协议中传输数据的基本单位。TCP协议基于流控制、顺序控制和错误校验等机制,以确保数据的可靠传输。TCP报文段结构由多个字段组成,每个字段在TCP的工作中都有特定的作用 一个典型的TCP报文段由两部分组成: TCP头部&a…...
USB 驱动开发 --- Gadget 驱动框架梳理(一)
本文由 Linux 内核文档翻译与总结而来,个人学习笔记仅供参考。 Gadget 框架 在 USB 协议交互过程中,角色定义: the device driver is the master (or “client driver”) Linux 内核中称为 HCD(Host Controller Driver),负责与 …...
C#读写ini配置文件保存设置参数
本示例使用设备:https://item.taobao.com/item.htm?spma21dvs.23580594.0.0.52de2c1b5P5rkA&ftt&id22173428704 [DllImport("kernel32", CharSet CharSet.Unicode)] public static extern uint GetPrivateProfileString(string lpAppName, stri…...
Linux---history查看命令历史记录命令
history命令是Linux系统中用于显示和管理用户之前执行过的命令列表的实用工具。以下是history命令的详细教程: 一、基本功能 history命令能够显示用户在当前shell会话或之前会话中执行过的所有命令。这对于回顾之前的操作、复制某个命令进行修改再次执行等场景非常…...
CANopen 学习笔记(2)
PDO通讯参数 异步传输 发送类型设置为0xFE或0xFF 修改映射参数后执行sendPDOevent(&CanFestival_Master_Data); /* 发送PDO事件 */则会触发PDO传输 还有一直就是定时器异步发送,配置发送类型为0xFE,定时器发送时间为0x64,则会以100ms的周期发送P…...
《鸿蒙Next旅游应用:人工智能赋能个性化与智能导览新体验》
随着鸿蒙Next的推出,旅游应用迎来了全新的发展机遇,借助人工智能技术能为用户带来更出色的个性化推荐和智能导览服务。 鸿蒙Next与人工智能融合优势 鸿蒙Next拥有强大的分布式能力和原生智能体验。其能打破设备界限,实现多设备协同…...
计算机网络 (39)TCP的运输连接管理
前言 TCP(传输控制协议)是一种面向连接的、可靠的传输协议,它在计算机网络中扮演着至关重要的角色。TCP的运输连接管理涉及连接建立、数据传送和连接释放三个阶段。 一、TCP的连接建立 TCP的连接建立采用三次握手机制,其过程如下&…...
Level2逐笔成交逐笔委托毫秒记录:今日分享优质股票数据20250114
逐笔成交逐笔委托下载 链接: https://pan.baidu.com/s/18YtQiLnt06cPQP1nRXor0g?pwd4k3h 提取码: 4k3h Level2逐笔成交逐笔委托数据分享下载 基于Level2的逐笔成交和逐笔委托数据,这种毫秒级别的记录能分析出许多关键信息,如庄家意图、虚假动作&#…...
探索Java(适合小白)
探索Java:推动数字世界的语言 Java,这一编程语言在信息技术的海洋中如同明珠般闪耀,吸引着无数程序员的关注与热爱。在今天的博客中,我们将深入了解Java的定义、学习的重要性,并详细探讨Java EE的概念。无论你是编程新…...
Java Web开发高级——Spring Boot与微服务架构
微服务架构(Microservices Architecture)已经成为现代软件开发中的主流架构之一。它通过将单一的、庞大的应用程序拆分成多个小的、独立部署的服务,使得开发、维护和扩展变得更加灵活、可控。Spring Boot 提供了一种简单且高效的方式来构建微…...
IMX6U Qt 开发环境
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言 一、交叉编译 1. 安装通用 ARM 交叉编译工具链 2. 安装 Poky 交叉编译工具链 二、编译出厂源码 1. U-boot 2. 内核和模块 3. 编译出厂 Qt GUI 综合 Demo 前言…...
【计算机网络】lab5 ARP协议
🌈 个人主页:十二月的猫-CSDN博客 🔥 系列专栏: 🏀计算机网络_十二月的猫的博客-CSDN博客 💪🏻 十二月的寒冬阻挡不了春天的脚步,十二点的黑夜遮蔽不住黎明的曙光 目录 1. 前言 2.…...
基于深度学习的视觉检测小项目(十三) 资源文件的生成和调用
在使用 PySide6 进行开发时,管理应用程序的资源(如图标、图片、字体、样式表、音视频等)是一个常见的任务。PySide6 提供了一个工具 pyside6-rcc,它能够将资源文件(.qrc)编译成 Python 模块,然后…...
【C】初阶数据结构3 -- 单链表
之前在顺序表那一篇文章中,提到顺序表具有的缺点,比如头插,头删时间复杂度为O(n),realloc增容有消耗等。而在链表中,这些问题将得到解决。所以在这一篇文章里,我们将会讲解链表的定义与性质,以及…...
STM32 FreeRTOS 基础知识
多任务处理 内核是操作系统的核心组件。诸如 Linux 这样的操作系统采用的内核, 看似允许用户同时访问计算机。很明显,多个用户可以同时执行多个程序。 每个执行程序都是受操作系统控制的任务(或线程)。如果一个操作系统能够以这…...
初学stm32 --- II2C_AT24C02,向EEPROM中读写数据
目录 IIC总线协议介绍 IIC总线结构图 IIC协议时序 1. ACK(Acknowledge) 2. NACK(Not Acknowledge) IO口模拟II2C协议 发送起始信号: 发送停止信号: 检测应答信号: 发送应答信号&#x…...
探索图像编辑的无限可能——Adobe Photoshop全解析
文章目录 前言一、PS的历史二、PS的应用场景三、PS的功能及工具用法四、图层的概念五、调整与滤镜六、创建蒙版七、绘制形状与路径八、实战练习结语 前言 在当今数字化的世界里,视觉内容无处不在,而创建和编辑这些内容的能力已经成为许多行业的核心技能…...
当comfyui-reactor-node 安装失败urllib.error.HTTPError: HTTP Error 403: Forbidden解决方法
comfyUI 节点comfyui-reactor-node 安装 python install 时 报错 urllib.error.HTTPError: HTTP Error 403: Forbidden 如下: (xxx) xxxxxxx:~/sdb/Q/ComfyUI/custom_nodes/comfyui-reactor-node$ python install.py Traceback (most recent call last): File …...
01基本介绍篇(D2_多线程问题)
目录 一、线程的上下文切换问题 1. 基本介绍 2. 多线程一定比单线程快? 3. 如何减少上下文切换 二、线程安全问题 1. 什么是线程安全? 2. java语言中的线程安全 2.1. 不可变 2.2. 绝对线程安全 2.3. 相对线程安全 2.4. 线程兼容 2.5. 线程对立…...
如何保证光谱相机的稳定性和可靠性
光学系统设计与制造 高质量光学元件:采用高精度研磨和镀膜的透镜、棱镜、光栅等光学元件。优质的透镜可以减少像差和色差,确保光线准确聚焦;高质量的镀膜能够提高光学元件的透光率,降低反射损失,并且增强对不同波段光…...
基于springboot+vue的洪涝灾害应急信息管理系统设计与实现
开发语言:Java框架:springbootJDK版本:JDK1.8服务器:tomcat7数据库:mysql 5.7(一定要5.7版本)数据库工具:Navicat11开发软件:eclipse/myeclipse/ideaMaven包:…...
登录、注册、忘记密码、首页HTML模板
<!DOCTYPE html> <html lang"zh"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>登录</title><style>body {display: fl…...
HTML文章翻页功能
效果展示: 效果原理: 1、引入CDN 2、绘制文章翻页样式,以及自动分段 3、获取窗口宽高,计算出当前文章总分段,并实现分页 4、完整代码 <!DOCTYPE html> <html><head><meta charset"utf-8&qu…...
JAVA安全编码规范
1. 数据校验 对外部输入进行校验入参的合法性, 防止内存越界,命令注入,SQL注入,格式化字符串漏洞 校验长度,范围,输入校验采用白名单形式 校验前做归一化处理,使用java.text.Normalizer的n…...
OpenGL —— 基于Qt的视频播放器 - ffmpeg硬解码,QOpenGL渲染yuv420p或nv12视频(附源码)
运行效果 工程说明 源码 vertex.glsl...
爬虫逆向学习(十五):Akamai 3.0反爬分析与sensor-data算法逆向经验
此分享只用于学习用途,不作商业用途,若有冒犯,请联系处理 Akamai 3.0反爬分析与sensor-data算法逆向经验 Akamai开始正题前须知站点信息接口分析反爬点反爬点定位_abck定位结果 逆向前准备工作sensor_data生成位置本地替换文件 请求体sensor…...
java项目启动时,执行某方法
1. J2EE项目 在Servlet类中重写init()方法,这个方法会在Servlet实例化时调用,即项目启动时调用。 import javax.servlet.ServletException; import javax.servlet.http.HttpServlet;public class MyServlet extends HttpServlet {Overridepublic void …...
学会使用开源软件jclasslib 字节码文件的组成 详解
应用场景 1 应用场景 2 学习路线 以正确的姿势打开文件 字节码文件的组成 玩转字节码常用工具 以正确的姿势打开文件 开源软件 jclasslib github 地址 https://github.com/ingokegel/jclasslib 工具使用 字节码文件的组成 基本信息 常量池 字段 方法 属性 详解 魔数 主副版…...
Flask表单处理与验证
Flask是一个轻量级的Python框架,它通过扩展库提供了对表单处理与验证的支持。WTForms是一个流行的Flask扩展库,用于创建和验证Web表单。它提供了一种声明式的方法来定义表单结构和验证逻辑,使得表单处理更为简洁和优雅。下面,我们…...
如何通俗易懂的理解 html js css
HTML、CSS 和 JavaScript 是构建网页的三大核心技术。为了通俗易懂地理解它们,我们可以用一个简单的比喻:**盖房子**。 --- ### 1. **HTML:房子的结构** HTML(HyperText Markup Language)就像房子的**骨架**。它定义…...
信凯科技业绩波动明显:毛利率远弱行业,资产负债率偏高
《港湾商业观察》施子夫 1月8日,深交所官网显示,浙江信凯科技集团股份有限公司(以下简称“信凯科技”)主板IPO提交注册。 自2022年递交上市申请,信凯科技的IPO之路已走过两年光景,尽管提交注册࿰…...
蓝牙BT04-A的使用与相关AT指令
一、AT指令没有返回的问题及解决方案 检查指令格式: 确认指令格式是否正确,包括特定的命令和结尾的回车换行符(n)。 检查TX/RX连接: 确认TX(发送)和RX(接收)线是否连接正…...
新手如何练习SQL?|掌握
对于新手想要练习SQL语句,可以从以下几个方面入手: 1. 建立理论基础 首先深入理解数据库的核心组件,包括数据库本身、其内部的各个表、表中的字段及其对应的数据类型(如字符串、整型、日期等),以及数据库…...
JavaScript宝典下
小哆啦闭关修炼已久,潜心攻读专业秘技,方才下山考研本欲大展宏图,怎奈山河虽壮志难酬,终是觉察考研无望。思来想去,不若弃考研之念,重拾敲代码之道,复盘前端奇术,以备闯荡职场江湖。…...
浅谈云计算12 | KVM虚拟化技术
KVM虚拟化技术 一、KVM虚拟化技术基础1.1 KVM虚拟化技术简介1.2 KVM虚拟化技术架构1.2.1 KVM内核模块1.2.2 用户空间工具(QEMU、Libvirt等) 二、KVM虚拟化技术原理2.1 硬件辅助虚拟化2.2 VMCS结构与工作机制 三、KVM虚拟化技术面临的挑战与应对策略3.1 性…...
Spring Boot 动态表操作服务实现
Spring Boot 动态表操作服务实现 Spring Boot 动态表操作服务实现1. 环境配置2. JdbcTemplate 的使用2.1 创建动态表2.2 动态添加字段2.3 动态删除字段2.4 动态修改字段类型2.5 删除表的方法实现 3. 小结3.1 可能的优化 Spring Boot 动态表操作服务实现 在现代的应用开发中&am…...
62_Redis服务器集群优化
Redis集群虽然具备高可用特性,且能实现自动故障恢复,但是如果使用不当,也会存在一些问题,总结如下。 集群完整性问题集群带宽问题数据倾斜问题客户端性能问题命令的集群兼容性问题Lua和事务问题1.集群完整性问题 在 Redis 集群的默认配置下,当节点检测到存在至少一个哈希…...
晨辉面试抽签和评分管理系统之九:随机编排考生的分组(以教师资格考试面试为例)
晨辉面试抽签和评分管理系统(下载地址:www.chenhuisoft.cn)是公务员招录面试、教师资格考试面试、企业招录面试等各类面试通用的考生编排、考生入场抽签、候考室倒计时管理、面试考官抽签、面试评分记录和成绩核算的面试全流程信息化管理软件。提供了考生…...
Linux Top 命令 load average 指标解读
前言 作为平台开发的同学,维护平台稳定性是我们最基本的工作职责,下面主要介绍下top 命令里 ,load average 这个指标如何去衡量机器负载程度。 概念介绍 load average 是系统在过去 1 分钟、5 分钟、15 分钟 的平均负载,它表示运…...
Nacos: 一个动态服务发现与配置管理平台
Nacos: 一个动态服务发现与配置管理平台 引言 在微服务架构日益普及的今天,服务之间的调用和配置管理变得越来越复杂。为了简化这一过程并提高开发效率,阿里巴巴推出了Nacos——一个易于使用的动态服务发现、配置管理和服务管理平台。 Nacos是什么&am…...
SpringBoot + 事务钩子函数
一、案例背景 拿支付系统相关的业务来举例。在支付系统中,我们需要记录每个账户的资金流水(记录用户A因为哪个操作扣了钱,因为哪个操作加了钱),这样我们才能对每个账户的账做到心中有数,对于支付系统而言&…...
OpenCV相机标定与3D重建(56)估计物体姿态(即旋转和平移)的函数solvePnPRansac()的使用
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 使用RANSAC方案从3D-2D点对应关系中找到物体的姿态。 cv::solvePnPRansac 是 OpenCV 中用于估计物体姿态(即旋转和平移)的…...
【JVM中的三色标记法是什么?】
JVM中的三色标记法是什么? 一、基本概念二、标记过程三、优势与问题四、漏标与多标的解决方案三色标记法(Tri-color Marking Algorithm)是Java虚拟机(JVM)中一种用于追踪对象存活状态的垃圾回收算法。 它基于William D. Hana和Mark S. McCulleghan在1976年提出的两色标记法…...
从0开始学习搭网站第二天
前言:今天比较惭愧,中午打铲吃了一把,看着也到钻二了,干脆顺手把这个赛季的大师上了,于是乎一直到网上才开始工作,同样,今天的学习内容大多来自mdn社区mdn 目录 怎么把文件上传到web服务器采用S…...
43.Textbox的数据绑定 C#例子 WPF例子
固定最简步骤,包括 XAML: 题头里引入命名空间 标题下面引入类 box和block绑定属性 C#: 通知的类,及对应固定的任务 引入字段 引入属性 属性双触发,其中一个更新block的属性 block>指向box的属性 从Textbo…...
钉钉实现第三方登录示例(重复回调问题解析)
钉钉作为专门为企业打造的沟通协助平台,包含的功能很多,考勤打卡,审批,日记,钉盘,钉邮等。基本满足了一些中小企业的大部分工作需求。因此对接钉钉的一些功能模块业务需求在开发中也是比较常见的。钉钉的开…...
Vue2+OpenLayers添加/删除点、点击事件功能实现(提供Gitee源码)
目录 一、案例截图 二、安装OpenLayers库 三、安装Element-UI 四、代码实现 4.1、添加一个点 4.2、删除所有点 4.3、根据经纬度删除点 4.4、给点添加点击事件 4.5、完整代码 五、Gitee源码 一、案例截图 可以新增/删除标记点,点击标记点可以获取到当前标…...
算法妙妙屋-------2..回溯的奇妙律动
回溯算法是一种用于系统性地搜索和解决问题的算法,它以深度优先搜索(DFS)为基础,用来探索所有可能的解决方案。通过递归地尝试候选解并在必要时回退(即“回溯”),它能够高效地解决许多涉及组合、…...