【2025版】Spring Boot面试题
文章目录
- 1. Spring, Spring MVC, SpringBoot是什么关系?
- 2. 谈一谈对Spring IoC的理解
- 3. @Component 和 @Bean 的区别?
- 4. @Autowired 和 @Resource 的区别?
- 5. 注入Bean的方法有哪些?
- 6. 为什么Spring 官方推荐构造函数注入?
- 7. Bean 的作用域有哪些?
- 8. Bean 是线程安全的吗?
- 9. Bean 的生命周期了解吗?
- 10. 如何解决 Spring 中的循环依赖问题?
- 11. @Lazy能解决循环依赖问题吗?
- 12. Spring 动态代理默认用哪一种?
- 13. Spring中拦截器和过滤器的区别?
- 14. Spring Boot的配置优先级
- 15. Spring Boot 自动配置如何实现的?
- 16. @PathVariable 和 @RequestParam 的区别?
- 17. Spring MVC的工作流程?
- 18. Spring Boot 支持哪些嵌入式 web 容器?
- 19. 介绍一下@SpringBootApplication注解
- 20. Spring Boot 常用的两种配置文件是什么?
- 21. 如何使用Spring Boot实现全局异常处理?
- 22. Spring 中如何实现定时任务?多节点重复执行如何避免?
- 23. 你的项目是如何统一返回结果的?
- 24. 什么是Spring Boot Starters?
- 25. Spring Boot 的主要优点?
- 26. Spring Boot 是如何通过 main 方法启动 web 项目的?
- 27. 如何在 Spring Boot 中读取配置信息?
- 28. Spring 事务中哪几种事务传播行为?
- 29. Spring 中BeanFactory 和 FactoryBean是什么?
- 30. ApplicationContext是什么?
- 31. Spring Boot 如何做请求参数校验?
- 32. Spring Boot 如何处理跨域请求?
- 33. Spring Boot 如何实现异步处理?
- 34. 说说对 Spring 中事件机制的理解?
- 35. Spring 中如何配置多数据源?
- 36. Spring 中有哪些设计模式?
- 37. Spring AOP 如何使用?
- 38. @Primary 和 @Qualifier注解的作用是什么?
- 39. @RequestBody和@ResponseBody注解的作用是什么?
1. Spring, Spring MVC, SpringBoot是什么关系?
Spring 包含了多个功能模块,Spring MVC是其中一个模块,专门处理Web请求。Spring Boot 只是简化了配置,如果需要构建 MVC 架构的 Web 程序,还是需要使用 Spring MVC 作为 MVC 框架,只是说 Spring Boot 简化了 Spring MVC 的很多配置,真正做到开箱即用。
2. 谈一谈对Spring IoC的理解
IoC(Inversion of Control:控制反转) 将原本在程序中手动创建对象的控制权,交由 Spring 框架来管理。
为什么叫控制反转?
控制
:指的是对象创建(实例化、管理)的权力
反转
:控制权交给外部环境(Spring 框架、IoC 容器)
3. @Component 和 @Bean 的区别?
- @Component 注解作用于类,而@Bean注解作用于方法。
- 当我们引用第三方库中的类需要装配到 Spring容器时,则只能通过 @Bean来实现。
4. @Autowired 和 @Resource 的区别?
@Autowired
属于 Spring 内置的注解,默认的注入方式为byType(根据类型进行匹配)
,也就是说会优先根据接口类型去匹配并注入 Bean (接口的实现类)。 当一个接口存在多个实现类的话,byType这种方式就无法正确注入对象了,因为这个时候 Spring 会同时找到多个满足条件的选择,默认情况下它自己不知道选择哪一个。这种情况下,注入方式会变为 byName(根据名称进行匹配)
,这个名称通常就是类名(首字母小写)。
通过 @Qualifier 注解可以来显式指定名称而不是依赖变量的名称。
@Resource
属于 JDK 提供的注解,默认注入方式为 byName
。如果无法通过名称匹配到对应的 Bean 的话,注入方式会变为byType
。
@Resource 有两个比较常用的属性:name(名称)、type(类型)。如果仅指定 name 属性则注入方式为byName,如果仅指定type属性则注入方式为byType,如果同时指定name 和type属性(不建议这么做)则注入方式为byType+byName。
5. 注入Bean的方法有哪些?
- 构造函数注入:通过类的构造函数来注入依赖项。
- Setter 注入:通过类的 Setter 方法来注入依赖项。
- Field(字段) 注入:直接在类的字段上使用注解(如 @Autowired 或 @Resource)来注入依赖项。
6. 为什么Spring 官方推荐构造函数注入?
依赖完整性
:确保所有必需依赖在对象创建时就被注入,避免了空指针异常的风险。不可变性
:有助于创建不可变对象,提高了线程安全性。初始化保证
:组件在使用前已完全初始化,减少了潜在的错误。测试便利性
:在单元测试中,可以直接通过构造函数传入模拟的依赖项,而不必依赖 Spring 容器进行注入。
7. Bean 的作用域有哪些?
singleton
: IoC 容器中只有唯一的 bean 实例。Spring 中的 bean 默认都是单例的,是对单例设计模式的应用。prototype
: 每次获取都会创建一个新的 bean 实例。也就是说,连续 getBean() 两次,得到的是不同的 Bean 实例。request
: 每一次 HTTP 请求都会产生一个新的 bean,该 bean 仅在当前 HTTP request 内有效。session
: 每一个 HTTP Session 会产生一个新的 bean,该 bean 仅在当前 HTTP session 内有效。
8. Bean 是线程安全的吗?
Spring 框架中的 Bean 是否线程安全,取决于其作用域和状态
。
几乎所有场景的 Bean 作用域都是使用默认的singleton ,重点关注 singleton 作用域即可。prototype 作用域下,每次获取都会创建一个新的 bean 实例,不存在资源竞争问题,所以不存在线程安全问题。
singleton 作用域
下,IoC 容器中只有唯一的 bean 实例,可能会存在资源竞争问题(取决于 Bean 是否有状态)。如果这个 bean 是有状态的话,那就存在线程安全问题(有状态 Bean 是指包含可变的成员变量的对象)。
9. Bean 的生命周期了解吗?
Bean的生命周期从Spring容器启动开始,首先根据配置或注解扫描获取Bean的定义信息,随后通过反射实例化对象并填充属性,完成依赖注入。如果Bean实现了诸如BeanNameAware等Aware接口,容器会在此阶段回调相关方法使其感知自身信息。接下来,BeanPostProcessor的postProcessBeforeInitialization方法被调用,执行初始化前的自定义逻辑,例如处理@PostConstruct注解的方法。随后容器触发初始化回调,包括InitializingBean接口的afterPropertiesSet方法或通过XML、注解定义的初始化方法。BeanPostProcessor的postProcessAfterInitialization在此之后执行,常见于生成AOP代理对象等增强处理。此时Bean已就绪,可被应用程序使用。当容器关闭时,销毁流程启动,依次执行@PreDestroy注解的方法、DisposableBean接口的destroy方法或配置的销毁方法,最终完成Bean的资源释放与生命周期终结。
10. 如何解决 Spring 中的循环依赖问题?
Spring通过三个缓存层级解决单例Bean的循环依赖问题:
缓存名称 | 描述 |
---|---|
singletonObjects | 一级缓存:存放完全初始化好的Bean(成品对象)。 |
earlySingletonObjects | 二级缓存:存放早期暴露的Bean(已实例化但未填充属性,未初始化)。 |
singletonFactories | 三级缓存:存放Bean的工厂对象(ObjectFactory),用于生成早期引用。 |
以A依赖B,B依赖A为例:
-
创建Bean A
-
实例化A:调用A的构造函数创建对象(此时对象未填充属性,未初始化)。
-
将A的工厂对象放入三级缓存(singletonFactories),用于后续生成早期引用。
-
填充A的属性:发现需要注入B。
-
-
创建Bean B
-
实例化B:调用B的构造函数创建对象。
-
将B的工厂对象放入三级缓存。
-
填充B的属性:发现需要注入A。
-
-
解决B对A的依赖
-
从三级缓存中获取A的工厂对象(singletonFactories),生成A的早期引用(通过getEarlyBeanReference方法)。
这一步是整合了第一步那个实例化但是没在任何缓存里的A对象,给他变成了半成品代理对象(如果A有被代理的话)
-
将A的早期引用存入二级缓存(earlySingletonObjects),并从三级缓存中删除A的工厂。
-
将A的早期引用注入到B中,完成B的属性填充和初始化。
-
将初始化后的B存入一级缓存(singletonObjects)。
-
-
完成A的创建
-
从一级缓存中获取已初始化的B,注入到A中。
-
完成A的属性填充和初始化。
-
将A存入一级缓存,并从二级缓存中删除A的早期引用。
-
我直接在实例化的时候判断A有没有代理,如果有的话,我直接把代理对象放到二级缓存里不行吗?这样就不用三级缓存了
代理对象应该在bean 实例化→属性填充→初始化
做完之后才去生成的(bean的生命周期),假设没有出现循环依赖,bean能通过正常的生命周期生成代理,我们直接在bean没完成初始化前就生成代理对象了,就打乱了bean的生命周期了。
通过三级缓存,可以推迟bean的早期引用暴露,也就是说,要不要提前生成代理对象这个事情,推迟到循环依赖真正发生的时候。如果真发生了循环依赖,B才会调用getEarlyBeanReference方法
生成A的代理,如果没循环依赖的话,在二级缓存正常放填充好属性的A对象的,就不用提前把A的代理放二级缓存了。
注意点:
- 仅支持单例Bean的循环依赖
原型(Prototype)作用域的Bean无法通过缓存解决循环依赖,Spring会直接抛出异常。 - 构造器注入无法解决循环依赖
如果循环依赖通过构造函数参数注入(而非Setter方法或字段注入),Spring无法提前暴露对象,会抛出BeanCurrentlyInCreationException。
解释:如果两个Bean都是原型模式的话,那么创建A1需要创建一个B1,创建B1的时候要创建一个A2,创建A2又要创建一个B2,创建B2又要创建一个A3,创建A3又要创建一个B3,循环依赖就没办法解决了。
如果A和B的依赖都是通过构造器注入,那连一个半成品对象都创建不出来,也没办法解决循环依赖
11. @Lazy能解决循环依赖问题吗?
在一定程度上是能解决的
-
Spring 创建 A 时,发现它依赖 B,但 B 被标记为 @Lazy。
-
不立即初始化 B,而是注入一个 B 的代理对象(由 Spring 动态生成)。
-
当 A 的方法首次调用 b.xxx() 时,代理对象才会触发 B 的实际初始化。
-
此时 B 初始化时再去注入 A,由于 A 已经存在,循环依赖被解开。
12. Spring 动态代理默认用哪一种?
Spring Boot 2.x 及以上版本默认启用了 proxyTargetClass=true,因此无论目标类是否实现接口,统一使用 CGLIB 生成代理。
动态代理的使用
13. Spring中拦截器和过滤器的区别?
过滤器(Filter)
是Servlet规范的一部分,其作用范围覆盖整个Web应用,能够处理所有进入Servlet容器的请求,例如修改请求参数、设置字符编码或实现全局安全控制。它的执行时机在请求到达DispatcherServlet之前,因此可以作用于静态资源等非Spring管理的请求。
拦截器(Interceptor)
是Spring MVC框架提供的组件,其实现依赖于HandlerInterceptor接口,通过Spring的配置类注册到拦截器链中。它的核心作用范围集中在Spring管理的控制器(Controller)层,能够在请求进入具体Controller方法前(preHandle)、方法执行后视图渲染前(postHandle)以及整个请求完成后(afterCompletion)这三个关键节点插入逻辑。所以拦截器更适合处理与业务紧密相关的操作,例如基于会话的权限校验、日志记录或接口性能监控。从执行顺序上看,整体上过滤器的处理会先于拦截器完成。过滤器更偏向底层请求的通用处理,而拦截器则聚焦于Spring MVC流程中的业务逻辑增强。
HTTP Request → Servlet Filter (过滤请求) → DispatcherServlet → Interceptor.preHandle() → Controller → Interceptor.postHandle() → Interceptor.afterCompletion()
14. Spring Boot的配置优先级
从高到低:
- 命令行参数(–key=value)
- java系统属性(-Dkey=value)
- application.properties
- application.yml
15. Spring Boot 自动配置如何实现的?
https://blog.csdn.net/fim77/article/details/146459033
16. @PathVariable 和 @RequestParam 的区别?
@PathVariable用于获取路径参数 /users/{id} → /users/123
@RequestParam用于获取查询参数。/users?id=123
17. Spring MVC的工作流程?
图片来自面试鸭用户:https://www.mianshiya.com/user/1815995005551374337
18. Spring Boot 支持哪些嵌入式 web 容器?
SpringBoot提供了三种内嵌Web容器,分别为Tomcat、Jetty和Undertow。
当你在项目中引入spring-boot-starter-web这个起步依赖时,Spring Boot默
认会包含并启用Tomcat作为内嵌Servlet容器。
如果你想使用Jetty或Undertow,需要在构建文件(如Maven的pom.xml或
Gradle 的build.gradle)中,从spring-boot-starter-web 中排除默认的
Tomcat 依赖(spring-boot-starter-tomcat),添加你想使用的容器对应的
Starter 依赖(例如spring-boot-starter-jetty 或spring-boot-starter-un
dertow)
19. 介绍一下@SpringBootApplication注解
@SpringBootApplication是SpringBoot项目的核心注解,通常用于标记应用程
序的主类(即包含main方法的类)。它的主要作用是一站式地启用SpringBoot的
关键特性,简化项目的初始配置。
@SpringBootConfiguration
继承自 @Configuration,标记当前类为配置类,允许通过 @Bean 注解定义和注册 Bean。-
@EnableAutoConfiguration
启用 Spring Boot 的自动配置机制。根据项目依赖中META-INF/下后缀为.imports 文件加载预定义的配置类,结合条件注解(如 @ConditionalOnClass)自动配置 Spring 应用所需的 Bean。 @ComponentScan
默认扫描当前类所在包及其子包下的组件,并将它们注册为 Spring Bean。
20. Spring Boot 常用的两种配置文件是什么?
application.properties
采用标准的JavaProperties文件格式,即键值对(key=value)的形式,每一行定义一个配置项。
server.port=8080
spring.datasource.url=jdbc:mysql://localhost:3306/mydb
application.yml(或yaml)
采用YAML(YAMLAin’tMarkupLanguage)格式,这是一种层级化、以缩进表示结构的数据序列化语言。相比.properties文件,YAML 格式通常更易于阅读,尤其是
在配置项较多或具有嵌套结构时,结构更清晰。并且,对于具有共同前缀的配置项,YAML可以通过层级嵌套避免重复书写前缀,使配置更简洁。
server:port: 8080
spring:datasource:url: jdbc:mysql://localhost:3306/mydb
21. 如何使用Spring Boot实现全局异常处理?
在 Spring Boot 中,可以通过 @ControllerAdvice
和 @ExceptionHandler
注解实现全局异常处理。
@RestControllerAdvice
public class GlobalExceptionHandler {// 处理自定义异常@ExceptionHandler(BusinessException.class)public ResponseEntity<ErrorResponse> handleBusinessException(BusinessException ex) {ErrorResponse error = new ErrorResponse(ex.getCode(), ex.getMessage());return ResponseEntity.status(ex.getHttpStatus()).body(error);}// 处理所有未捕获的异常@ExceptionHandler(Exception.class)public ResponseEntity<ErrorResponse> handleGenericException(Exception ex) {ErrorResponse error = new ErrorResponse("ERROR_500", "系统内部错误");return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(error);}
}
22. Spring 中如何实现定时任务?多节点重复执行如何避免?
在 Spring Boot 主类或配置类上添加 @EnableScheduling
@SpringBootApplication
@EnableScheduling
public class Application {public static void main(String[] args) {SpringApplication.run(Application.class, args);}
}
使用 @Scheduled
注解标记方法,支持 cron、fixedRate、fixedDelay 等参数
@Component
public class MyScheduledTasks {// 每 5 秒执行一次@Scheduled(fixedRate = 5000)public void doTask() {System.out.println("执行定时任务: " + new Date());}
}
Spring Task 在多节点部署时,如果不采取措施,每个节点都会执行相同的定时任务,导致重复执行。这主
要是因为每个节点上都运行着独立的 Spring 容器,每个容器都拥有自己的定时任务调度器,并独立地根据
配置的时间触发任务,互不干扰。如果多个节点的配置相同,就会导致同一任务在多个节点上并发执行。
这种情况不仅浪费资源,还可能导致数据不一致、资源竞争等问题,最终导致业务逻辑错误,例如重复处
理相同的数据、发送重复的通知。
解决方法:
- 分布式锁
- 分布式任务调度工具,如XXL-JOB
23. 你的项目是如何统一返回结果的?
code
: 状态码,遵循HTTP状态码规范并扩展业务状态码
message
: 对状态的描述信息
data
: 实际返回的业务数据
timestamp
: 响应时间戳
- 手动显式封装
@GetMapping("/{id}")public Result<User> getUser(@PathVariable Long id) {User user = userService.getUserById(id);return Result.success(user);}
- 自动封装
@RestControllerAdvice
public class ResponseAdvice implements ResponseBodyAdvice<Object> {@Overridepublic boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {// 判断是否需要包装return !returnType.getParameterType().equals(Result.class);}@Overridepublic Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType,Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {return Result.success(body);}
}
24. 什么是Spring Boot Starters?
Spring Boot Starters 是一组便捷的依赖描述符,它们预先打包了常用的库和配置。当我们开发 Spring 应
用时,只需添加一个 Starter 依赖项,即可自动引入所有必要的库和配置,而无需手动逐一添加和配置相关
依赖。
这种机制显著简化了开发过程,特别是在处理复杂项目时尤为高效。通过添加一个简单的Starter 依赖,开
发者可以快速集成所需的功能,避免了手动管理多个依赖的繁琐和潜在错误。这不仅节省了时间,还减少
了配置错误的风险,从而提升了开发效率。
25. Spring Boot 的主要优点?
-
显著提升开发效率
:Spring Boot 通过自动配置、起步依赖(Starters)和其他开箱即用的功能,极大地减少了项目初始化、配置编写和样板代码的工作量,使开发者能更快地构建和交付应用。 -
与 spring 生态系统的无缝集成
:作为 Spring 家族的一员,Spring Boot 能够方便地整合 Spring 框架下的其他成熟模块(如 Spring Data、Spring Security、Spring Batch 等),充分利用 Spring 强大的生态系统,简化整合工作。 -
强大的自动配置能力
:遵循“约定优于配置”的原则,Spring Boot 能够根据项目依赖自动配置大量
的常见组件(如数据源、Web 容器、消息队列等),提供合理的默认设置。同时也允许开发者根据需
要轻松覆盖或定制配置,极大减少了繁琐的手动配置工作。 -
内嵌 Web 服务器支持
:Spring Boot 自带内嵌的 HTTP服务器(如 Tomcat、Jetty),开发者可以像运行普通 Java 程序一样运行 Spring Boot 应用程序,极大地简化了开发和测试过程。 -
适合微服务架构
:Spring Boot 使得每个微服务都可以独立运行和部署,简化了微服务的开发、测试和运维工作,成为构建微服务架构的理想选择。 -
提供强大的构建工具支持
:Spring Boot为常用的构建工具(如 Maven和 Gradle)提供了专门的插件,简化了项目的打包(如创建可执行JAR)、运行、测试以及依赖管理等常见构建任务丰富的监控和管理功能:通过 Spring Boot Actuator 模块,可以轻松地为应用添加生产级的监控和管理端点,方便了解应用运行状况、收集指标、进行健康检查等。
26. Spring Boot 是如何通过 main 方法启动 web 项目的?
https://www.mianshiya.com/bank/1797452903309508610/question/1846441429268488194#heading-14
27. 如何在 Spring Boot 中读取配置信息?
- 使用
@Value
注解
// 直接注入配置项,支持默认值(如未配置则使用默认值)@Value("${book.author:defaultAuthor}")private String author;
- 使用
@ConfigurationProperties
注解
# 配置文件 application.yml
book:name: 三国演义author: 罗贯中price: 30chapters:- 第一章- 第二章
@Component
@ConfigurationProperties(prefix = "book")
public class BookProperties {private String name;private String author;private int price;private List<String> chapters;
}
- 使用
Environment
接口
@Autowiredprivate Environment env;public String getAppInfo() {String appName = env.getProperty("app.name");String maxRetry = env.getProperty("app.max-retry", "defaultRetry");}
28. Spring 事务中哪几种事务传播行为?
总共有7种,常用的就前两种
PROPAGATION_REQUIRED(默认)
如果当前存在事务,则加入该事务;否则新建一个事务。
所有嵌套方法共享同一个事务,任一方法抛出异常会导致整个事务回滚。PROPAGATION_REQUIRES_NEW
无论当前是否存在事务,都新建一个独立事务。
内层事务与外层事务完全隔离,外层事务回滚不影响内层事务。PROPAGATION_NESTED
如果当前存在事务,则在当前事务内嵌套一个子事务;否则创建新事务。
子事务是父事务的一部分,但可以独立回滚。
子事务回滚不影响父事务(需捕获异常),父事务回滚则子事务也会回滚。PROPAGATION_MANDATORY
必须在一个已存在的事务中运行,否则抛出异常。PROPAGATION_SUPPORTS
如果当前存在事务,则加入该事务;否则以非事务方式运行。PROPAGATION_NOT_SUPPORTED
以非事务方式运行,如果当前存在事务则挂起。PROPAGATION_NEVER
以非事务方式运行,如果当前存在事务则抛出异常。
29. Spring 中BeanFactory 和 FactoryBean是什么?
BeanFactory
是Spring框架的核心接口,作为IoC容器的基础,负责管理Bean的生命周期,包括创建、配置和装配对象。
FactoryBean
是一个特殊的Bean,可以在运行时根据特定条件或需求,通过getObject()方法动态控制Bean的实例化过程,在容器中通过名称前加&符号区分获取FactoryBean本身与其生产的对象。
30. ApplicationContext是什么?
ApplicationContext是Spring框架的核心容器接口,它作为高级容器不仅继承了BeanFactory的基础功能来管理和配置Bean,还扩展了众多企业级特性。它整合了资源加载、国际化支持、事件发布与监听机制,同时支持AOP、事务管理等高级功能。
31. Spring Boot 如何做请求参数校验?
验证请求体
在 DTO 类上使用校验注解
public class UserDTO {@Size(min = 2, max = 20, message = "用户名长度必须在2-20之间")private String username;
}
在 Controller 中使用 @Valid 注解触发校验(@Validated也行)
@PostMapping("/users")
public ResponseEntity<?> createUser(@RequestBody @Valid UserDTO userDTO) {// 业务逻辑return ResponseEntity.ok("用户创建成功");
}
验证请求参数
直接在方法参数上使用校验注解:
@GetMapping("/users/{id}")
public ResponseEntity<?> getUserById(@PathVariable @Min(1) Long id) {// 业务逻辑return ResponseEntity.ok(...);
}
需要在 Controller 类上添加 @Validated 注解
,只能在类上添加这个才有用
@RestController
@Validated
public class UserController {// ...
}
补充:分组校验
public class UserDTO {@NotBlank(message = "用户名不能为空", groups = {CreateGroup.class, UpdateGroup.class})private String username;@Email(message = "邮箱格式不正确", groups = CreateGroup.class) // 仅创建时需要校验邮箱private String email;@NotNull(message = "ID不能为空", groups = UpdateGroup.class) // 仅更新时需要校验IDprivate Long id;// Getter 和 Setter
}
@PostMapping("/users")
public ResponseEntity<?> createUser(@RequestBody @Validated(CreateGroup.class) UserDTO userDTO) {return ResponseEntity.ok("创建用户校验通过");
}
32. Spring Boot 如何处理跨域请求?
一文搞懂Spring Boot处理跨域请求
33. Spring Boot 如何实现异步处理?
主要有四种方式来实现异步处理:
- 使用 @Async注解
- 使用 CompletableFuture
- 使用@Scheduled注解
- 使用线程池
34. 说说对 Spring 中事件机制的理解?
Spring 的事件机制是一种基于观察者模式设计的解耦通信方式,允许组件在特定动作发生时通过事件传递信息,而不必直接依赖彼此。开发者可以自定义继承 ApplicationEvent 的事件类,由事件发布者通过 ApplicationEventPublisher 触发事件,而监听者通过实现 ApplicationListener 接口或使用 @EventListener 注解来捕获并处理事件。这种机制默认以同步方式运行,但结合 @Async 或自定义线程池可支持异步处理,提升系统响应能力。例如,用户注册成功后发布事件,由监听器异步发送邮件或初始化数据,避免主流程阻塞。
35. Spring 中如何配置多数据源?
一文带你理清SpringBoot如何实现多数据源动态路由【全干货,无废话】
36. Spring 中有哪些设计模式?
工厂模式(Factory Pattern)
作用:隐藏对象创建细节,由容器统一管理 Bean 的生命周期。单例模式(Singleton Pattern)
Spring 默认的 Bean 作用域为单例(singleton),确保每个容器中一个 Bean 仅有一个实例。
作用:节省资源,保证全局一致性。原型模式(Prototype Pattern)
Bean 的作用域设为 prototype 时,每次请求都会创建新实例(如 @Scope(“prototype”))。代理模式(Proxy Pattern)
Spring AOP(面向切面编程)通过动态代理实现横切关注点(如事务、日志)。模板方法模式(Template Method Pattern)
JdbcTemplate、RestTemplate 等模板类封装通用流程,消除重复代码观察者模式(Observer Pattern)
核心应用:Spring 事件驱动模型(ApplicationEvent 和 ApplicationListener)。适配器模式(Adapter Pattern)
统一接口,兼容不同实现。装饰者模式(Decorator Pattern)
HttpServletRequestWrapper 包装 HTTP 请求,增强其行为(如缓存请求体)。策略模式(Strategy Pattern)
Spring MVC 的 HandlerMapping 根据请求匹配不同策略的处理器。委派模式(Delegate Pattern)
DispatcherServlet 将请求分发给不同的处理器(如 Controller、Handler)。建造者模式(Builder Pattern)
分步构建复杂对象。责任链模式(Chain of Responsibility)
Spring Security 的过滤器链(FilterChainProxy),每个过滤器依次处理请求。
37. Spring AOP 如何使用?
【25年最新AOP入门】最干净省时的SpringBoot AOP入门教程,一小时带你学会面向切面编程怎么用
38. @Primary 和 @Qualifier注解的作用是什么?
@Primary
标记某个Bean为“默认首选”的候选对象。当存在多个相同类型的Bean时,Spring会优先选择带有@Primary注解的Bean进行注入。
@Qualifier
通过显式指定 Bean 的名称或标识符来解决多个同类型 Bean 的冲突。
39. @RequestBody和@ResponseBody注解的作用是什么?
@RequestBody
注解用于将HTTP请求体中的内容绑定到方法的参数上,它主要作用在方法的参数上。
@ResponseBody
注解用于将方法的返回值直接写入HTTP响应体中,它可以标注在方法或类上。
在Spring 4.0之后,可以使用@RestController注解替代@Controller+@ResponseBody的组合
相关文章:

【2025版】Spring Boot面试题
文章目录 1. Spring, Spring MVC, SpringBoot是什么关系?2. 谈一谈对Spring IoC的理解3. Component 和 Bean 的区别?4. Autowired 和 Resource 的区别?5. 注入Bean的方法有哪些?6. 为什么Spring 官方推荐构造函数注入?…...

火山引擎实时音视频 高代码跑通日志
实时音视频 SDK 概览--实时音视频-火山引擎 什么是实时音视频 火山引擎实时音视频(Volcengine Real Time Communication,veRTC)提供全球范围内高可靠、高并发、低延时的实时音视频通信能力,实现多种类型的实时交流和互动。 通…...
atoi函数,sprintf函数,memcmp函数,strchar函数的具体原型,功能,返回值;以及使用方法
以下是这四个C语言标准库函数的详细说明: 1. atoi() - 字符串转整数 **原型**: int atoi(const char *str); **功能**: 将字符串参数str转换为整数(int类型)。函数会跳过前面的空白字符(如空格、制表符&am…...
C++学习之打车软件git版本控制
目录 01 3-git的简介 02 4-git的下载和提交代码 03 5-git添加一个新文件 04 5-删除一个文件 05 6-git的批量添加和提交文件 06 7-git重命名文件名 07 8-git解决代码冲突 08 9-git的分支的概念 09 10-创建项目代码仓库 10 1-git提交代码复习 01 3-git的简介 1 --------…...
基于 PostgreSQL 的 ABP vNext + ShardingCore 分库分表实战
🚀 基于 PostgreSQL 的 ABP vNext ShardingCore 分库分表实战 📑 目录 🚀 基于 PostgreSQL 的 ABP vNext ShardingCore 分库分表实战✨ 背景介绍🧱 技术选型🛠️ 环境准备✅ Docker Compose(多库 & 读…...

jenkins 启动报错
java.lang.UnsatisfiedLinkError: /opt/application/jdk-17.0.11/lib/libfontmanager.so: libfreetype.so.6: cannot open shared object file: No such file or directory。 解决方案: yum install freetype-devel 安装完成之后重启jenkins。...
C++ 套接字函数详细介绍
目录 头文件1. 套接字创建与配置2. 绑定地址与端口3. 连接建立4. 数据传输5. 套接字选项6. 地址转换7. 套接字关闭8. 其他实用函数 C 套接字函数详细介绍 套接字(Socket)是网络通信的基本端点,C中通常使用BSD套接字API进行网络编程。以下是主要的套接字相关函数及其…...

【合新通信】无人机天线拉远RFOF(射频光纤传输)解决方案
无人机天线拉远RFOF方案通过光纤替代传统射频电缆,实现无人机与地面控制站之间的高保真、低损耗信号传输,尤其适用于高频段(如毫米波)、远距离或复杂电磁环境下的无人机作业场景。 核心应用场景 军事侦察与电子战 隐蔽部署&…...

程序设计语言----软考中级软件设计师(自用学习笔记)
目录 1、解释器和编译器 2、程序的三种控制结构 3、程序中的数据必须具有类型 4、编译、解释程序翻译阶段 5、符号表 6、编译过程 7、上下文无关文法 8、前、中、后缀表达式 9、前、后缀表达式计算 10、语法树中、后序遍历 11、脚本语言和动态语言 12、语法分析方法…...
火山RTC 7 获得远端裸数据
一、获得远端裸数据 1、获得h264数据 1)、远端编码后视频数据监测器 /*** locale zh* type callback* region 视频管理* brief 远端编码后视频数据监测器<br>* 注意:回调函数是在 SDK 内部线程(非 UI 线程)同步抛出来的&a…...

通过SMTP协议实现Linux邮件发送配置指南
一、环境准备与基础配置 1. SMTP服务开通(以qq邮箱为例) 登录qq邮箱网页端,进入「设置」-「POP3/SMTP/IMAP」 开启「SMTP服务」并获取16位授权码(替代邮箱密码使用) 记录关键参数: SMTP服务器地址&#…...

若依框架页面
1.页面地址 若依管理系统 2.账号和密码 管理员 账号admin 密码admin123 运维 账号yuwei 密码123456 自己搭建的地址方便大家学习,不要攻击哦,谢谢啊...

44、私有程序集与共享程序集有什么区别?
私有程序集(Private Assembly)与共享程序集(Shared Assembly)是.NET框架中程序集部署的两种不同方式,它们在部署位置、版本控制、访问权限等方面存在显著差异,以下是对二者的详细比较: 1. 部署…...

【Java面试题】——this 和 super 的区别
🎁个人主页:User_芊芊君子 🎉欢迎大家点赞👍评论📝收藏⭐文章 🔍系列专栏:【Java】内容概括 【前言】 在Java的世界里,this和 super是两个非常重要且容易混淆的关键字。无论是在日常…...
记录 QT 在liunx 下 QFileDialog 类调用问题 ()Linux下QFileDialog没反应)
1. 2025.05.14 踩坑记录 因为QT 在 liunx 文件系统不同导致的 Windows : QString filePath QFileDialog::getOpenFileName(nullptr, "选择文件", ".", "文本文件 (*.txt);所有文件 (*.*)"); 没问题 liunx 下 打不开ÿ…...

CentOS 7 内核升级指南:解决兼容性问题并提升性能
点击上方“程序猿技术大咖”,关注并选择“设为星标” 回复“加群”获取入群讨论资格! CentOS 7 默认搭载的 3.10.x 版本内核虽然稳定,但随着硬件和软件技术的快速发展,可能面临以下问题: 硬件兼容性不足:新…...
【前端】:单 HTML 去除 Word 批注
在现代办公中,.docx 文件常用于文档编辑,但其中的批注(注释)有时需要在分享或归档前被去除。本文将从原理出发,深入剖析如何在纯前端环境下实现对 .docx 文件注释的移除,并提供完整的实现源码。最后&#x…...

解决 PicGo 上传 GitHub图床及Marp中Github图片编译常见难题指南
[目录] 0.行文概述 1.PicGo图片上传失败 2.*关于在Vscode中Marp图片的编译问题* 3.总结与启示行文概述 写作本文的动机是本人看到了Awesome Marp,发现使用 Markdown \texttt{Markdown} Markdown做PPT若加持一些 CSS , JavaScript \texttt{CSS},\texttt{JavaScript} …...

软考 系统架构设计师系列知识点之杂项集萃(59)
接前一篇文章:软考 系统架构设计师系列知识点之杂项集萃(58) 第96题 假设关系模式R(U, F),属性集U{A, B, C},函数依赖集F{A->B, B->C}。若将其分解为p{R1(U1, F1), R2(U2, F2),其中U1{A, B}, U2{A, …...

python使用matplotlib画图
【README】 plot画图有两种方法:包括 plt.plot(), ax.plot()-画多个子图 ,其中ax表示某个坐标轴; 【1】画单个图 import matplotlib # 避免兼容性问题:明确指定 matplotlib 使用兼容的后端,TkAgg 或 Agg matplotlib.use(TkAgg) …...
鸿蒙OSUniApp 开发实时聊天页面的最佳实践与实现#三方框架 #Uniapp
使用 UniApp 开发实时聊天页面的最佳实践与实现 在移动应用开发领域,实时聊天功能已经成为许多应用不可或缺的组成部分。本文将深入探讨如何使用 UniApp 框架开发一个功能完善的实时聊天页面,从布局设计到核心逻辑实现,带领大家一步步打造专…...

upload-labs通关笔记-第5关 文件上传之.ini绕过
目录 一、ini文件绕过原理 二、源码审计 三、渗透实战 1、查看提示 2、制作.user.ini文件 (1)首先创建一个文本文件 (2)保存文件名为.user.ini 2、制作jpg后缀脚本 (1)创建一个文本文件 …...

ssti模板注入学习
ssti模板注入原理 ssti模板注入是一种基于服务器的模板引擎的特性和漏洞产生的一种漏洞,通过将而已代码注入模板中实现的服务器的攻击 模板引擎 为什么要有模板引擎 在web开发中,为了使用户界面与业务数据(内容)分离而产生的&…...

填涂颜色(bfs)
归纳编程学习的感悟, 记录奋斗路上的点滴, 希望能帮到一样刻苦的你! 如有不足欢迎指正! 共同学习交流! 🌎欢迎各位→点赞 👍+ 收藏⭐ + 留言📝 含泪播种的人一定能含笑收获! 题目描述 由数字 0 0 0 组成的方阵中,有一任意形状的由数字 1 1 1 构成的闭合圈。现…...

ros1+docker环境快速搭建
快速使用python 解析ros1的bag消息ros这个东西可以说安装起来非常麻烦的,费时费力,很可能还安装不成功,特别是我的环境是ubuntu22.04 ,官方都不支持安装ros1。因此一个可行且快速的方法是使用别人配置好的ros的docker环境 一、下…...
vscode extention踩坑记
# npx vsce package --allow-missing-repository --no-dependencies #耗时且不稳定 npx vsce package --allow-missing-repository #用这行 code --install-extension $vsixFileName --force我问ai:为什么我的.vsix文件大了那么多 ai答:因为你没有用 --n…...

GpuGeek全栈AI开发实战:从零构建企业级大模型生产管线(附完整案例)
目录 背景一、算力困境:AI开发者的「三重诅咒」1.1 硬件成本黑洞1.2 资源调度失衡1.3 环境部署陷阱 二、三大核心技术突破GpuGeek的破局方案2.1 分时切片调度引擎(Time-Slicing Scheduler)2.2 异构计算融合架构2.3 AI资产自动化…...
【TroubleShoot】禁用Unity Render Graph API 兼容模式
使用Unity 6时新建了项目,有一个警告提示: The project currently uses the compatibility mode where the Render Graph API is disabled. Support for this mode will be removed in future Unity versions. Migrate existing ScriptableRenderPasses…...
数据库字段唯一性修复指南:从设计缺陷到规范实现
数据库字段唯一性修复指南:从设计缺陷到规范实现 一、问题背景 表结构设计缺陷: sys_user 表未对 dingtalk_user_id(钉钉用户ID)字段设置唯一性约束,导致数据重复,引发以下问题: 系统稳定性风…...

DataX从Mysql导数据到Hive分区表案例
0、下载DataX并解压到对应目录 DataX安装包,开箱即用,无需配置。 https://datax-opensource.oss-cn-hangzhou.aliyuncs.com/202308/datax.tar.gz 相关参考文档 https://github.com/alibaba/DataX/blob/master/hdfswriter/doc/hdfswriter.md 1、Hive分区…...