一、SpringBoot核心原理相关面试题
1、SpringBoot自动装配原理?
在SpringBoot的项目中的引导类有一个注解,@SpringBootApplication,这个注解是对三个注解进行了封装,分别是
- @SpringBootConfiguration:@SpringBootConfiguration是SpringBoot框架中的一个核心注解,用于表示当前类为SpringBoot应用的配置类。
- @EnableAutoConfiguration:是实现自动化配置的核心注解,该注解通过@Import注解导入对应的配置选择器。关键就是内部就是读取了该项目和该项目引用的jar包中的classpath路径下的MATE-INF下spring.factories文件中所配置类的全类名,在这些配置类中所定义的bean会根据条件注解所指定的条件来决定是否需要将其导入到Spring容器中。条件判断会像@ConditionalOnClass这样的注解,判断是否有对应的class文件,如果有则加载该类,把这个配置类的所有Bean放入到Spring容器中使用
- @ComponentScan用于扫描指定包及其子包下的组件,并将其注册为Spring容器中bean。
2、Spring的bean的生命周期
这个步骤还是挺多的,我看过一些源码,大概流程是这样的:
首先第一步实例化会通过一个非常重要的类叫BeanDefintion获取Bean的定义信息,里面封装了Bean的所有信息,比如类的全路径、是否延迟加载、是否为单例等这些信息。通过反射机制实现构造函数实例化Bean。
第二步:Bean的依赖注入,比如一些Set方法注入,像平时开发用的@Autowired都是这一步完成的
第三步:处理Aware接口,比如某一个bean实现了Aware接口就会重写这个方法执行
第四步:Bean的前置处理器执行BeanPostprocessor前置处理
第五步:初始化方法,实现了接口InitializingBean或自定义方法Init-method标签或 @PostConstruct
第六步:执行了Bean的后置处理器BeanPostprocessor后置处理,主要对Bean肯呢个增强,可能产生代理对象,之后就可以正常使用。
第七步:使用Bean对象完成后销毁。
3、Bean的作用域有哪些
(1) Singleton (单例)-默认作用域
定义:在整个Spring IoC容器中,对于每个Bean ID,只存在一个共享的Bean实例。
生命周期:容器启动时创建(如果设置为懒加载则在第一次请求时创建),容器关闭时销毁。
适用场景:无状态的Bean,例如工具类(Service层、DAO层等通常使用此作用域)
-
配置方式:
-
XML:
<bean id="..." class="..." scope="singleton"/>
-
注解:
@Scope("singleton")
或@Scope(ConfigurableBeanFactory.SCOPE_SINGLETON)
-
(2) Prototype(原型)
-
定义:每次通过容器请求(查找或注入)该Bean时,都会创建一个新的实例。
-
生命周期:实例化后交由客户端(调用者)管理,容器不再跟踪其生命周期,也不会负责销毁它。客户端需要自己清理原型作用域的对象并释放资源。
-
适用场景:需要保持状态的Bean,例如每次请求都需要一个新对象的场景(如购物车)。
-
配置方式:
-
XML:
<bean id="..." class="..." scope="prototype"/>
-
注解:
@Scope("prototype")
或@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
-
(3) Request(请求)
-
定义:每个HTTP请求都会创建一个新的Bean实例。该作用域仅在Web应用中有效。
-
生命周期:在一次HTTP请求内有效,请求结束后,Bean实例被销毁。
-
适用场景:用于存储与一次请求相关的数据,如表单数据、请求参数等。
-
配置方式:
-
XML:
<bean id="..." class="..." scope="request"/>
-
注解:
@RequestScope
或@Scope("request")
或@Scope(WebApplicationContext.SCOPE_REQUEST)
-
-
注意:需要Web感知的ApplicationContext支持。
(4) Session(会话)
-
定义:每个HTTP Session会创建一个新的Bean实例。该作用域仅在Web应用中有效。
-
生命周期:在一个用户会话(Session)内有效,会话超时或失效后,Bean实例被销毁。
-
适用场景:用于存储用户级别的信息,如用户登录状态、购物车内容等。
-
配置方式:
-
XML:
<bean id="..." class="..." scope="session"/>
-
注解:
@SessionScope
或@Scope("session")
或@Scope(WebApplicationContext.SCOPE_SESSION)
-
-
注意:需要Web感知的ApplicationContext支持。
(5) Application(应用)
-
定义:整个Web应用中只创建一个Bean实例。它类似于
singleton
,但不同之处在于:-
singleton
是每个Spring容器一个实例(一个应用中可以有多个容器)。 -
application
是每个ServletContext
一个实例(一个应用通常只有一个ServletContext
)。
-
-
生命周期:在
ServletContext
生命周期内有效,应用关闭时销毁。 -
适用场景:用于存储全局的、应用级别的信息,例如应用启动时加载的配置信息。
-
配置方式:
-
XML:
<bean id="..." class="..." scope="application"/>
-
注解:
@ApplicationScope
或@Scope("application")
或@Scope(WebApplicationContext.SCOPE_APPLICATION)
-
-
注意:需要Web感知的ApplicationContext支持。
(6) Websocket(WebSocket会话)
-
定义:每个WebSocket会话会创建一个新的Bean实例。该作用域仅在Web应用中有效。
-
生命周期:在一个WebSocket会话内有效,会话结束后,Bean实例被销毁。
-
适用场景:用于在WebSocket通信中保持状态,例如在聊天应用中为每个连接保存用户消息状态。
-
配置方式:
-
注解:
@Scope(scopeName = "websocket")
-
如何选择和作用域配置示例
选择建议:
-
绝大多数情况使用
singleton
:这是默认且最高效的选择,适用于无状态的服务组件。 -
需要状态时使用
prototype
:当你的对象需要保持可变状态,且不希望在不同调用间共享时使用。 -
Web相关数据使用
request
/session
/application
:根据数据的生命周期(一次请求、一次会话、整个应用)来选择。
配置示例(使用注解):
@Service
@Scope("singleton") // 可省略,因为默认就是singleton
public class DefaultService {
// ...
}
@Component
@Scope("prototype")
public class ShoppingCart {
// ...
}
@Controller
@RequestScope // 或 @Scope("request")
public class LoginController {
// ...
}
@Service
@SessionScope // 或 @Scope("session")
public class UserPreferences {
// ...
}
@Configuration
@ApplicationScope // 或 @Scope("application")
public class AppConfig {
// ...
}
(7)自定义作用域
除了上述标准作用域,Spring还提供了扩展机制,允许你注册和定义自己的自定义作用域(例如,为每个线程创建一个实例的“thread”作用域)。这需要通过org.springframework.beans.factory.config.Scope
接口来实现,并将其注册到容器中。
总结:
作用域 | 说明 | 适用场景 | 是否Web相关 |
---|---|---|---|
singleton | 容器中只有一个实例(默认) | 无状态服务,工具类 | 否 |
prototype | 每次请求都创建一个新实例 | 有状态对象 | 否 |
request | 一次HTTP请求一个实例 | 表单数据,请求参数 | 是 |
session | 一个用户会话一个实例 | 用户登录信息,购物车 | 是 |
application | 整个Web应用一个实例 | 全局配置,应用级缓存 | 是 |
websocket | 一个WebSocket会话一个实例 | WebSocket通信状态 | 是 |
4、Spring框架中的单例bean是线程安全的吗?
不是线程安全的,当多个用户同时请求一个服务时,容器会给每一个请求与分配一个线程,这时多个线程会并发处理对应的请求,如果处理逻辑对单例状态就行修改(修改单例的成员属性),必须考虑线程的同步问题。
Spring框架的单例Bean本身只是一个管理实例创建的模式,它并不保证线程安全。线程安全完全取决于开发者如何编写Bean内部的代码。对于无状态的Bean(例如只包含注入的Service和Dao),它是线程安全的;而对于包含可变状态的有状态Bean,则需要开发者自己通过同步机制(如锁、原子变量、ThreadLocal)或改变Bean作用域(将多态的Bean从singleton变更为prototype)来保证线程安全。
二、Spring应用方面面试题
1、Spring常见的注解有哪些(Spring、Spring MVC、Springboot)?
(1)Spring
- @Component、@Controller、@Service、@Repository (在类上实例化Bean的注解)
- @Autowired(使用字段上根据类型进行依赖注入)
- @Qualifier(结合@Autowired根据名称进行依赖注入)
- @Configuration(指定当前类是一个Spring配置类,当创建容器时会从该类上加载注解)
- @ComponentScan (用于指定Spring初始化容器时要扫描的包)
- @Bean 使用在方法上标注该方法返回值注入到容器中,经常与一个配置类并用
@Configuration搭配使用。
- @Import(使用@Import导入的类会被注册到Spring IOC的容器中)
- @Aspect、@Before、@After、@Around、@Pointcut(用于切面编程)
(2)Spring MVC
@RestController:
@Controller
和@ResponseBody组合注解
@RequestMapping:
通用的请求映射注解,用于将HTTP请求映射到特定的处理方法上。可以定义在类上(类中所有方法都是以改地址为父路径)和方法上@PathVariable:从请求路径下获取请求参数{/user/{id}}
@RequestBody:接收http请求的json数据,将JSON数据转化为java对象
@RequestHeader:
用于获取HTTP请求头部的值。@ResponseBody:将Controller方法返回的对象转化为JSON对象相应给客户端
-
@RequestParam
:用于获取URL查询参数或表单数据。
(3)SpringBoot
@SpringBootApplication:
这是一个组合注解,是启动 Spring Boot 应用的唯一入口-
@SpringBootConfiguration
:组合了@Configuration注解,表明该类是一个 Spring Boot 的配置类。 -
EnableAutoConfiguration:
显式启用自动配置 -
@ComponentScan Spring组件扫描
2、SpringMVC工作流程原理?
- 用户发出请求到前端控制器DispatcherServlet
- DispatcherServlet收到请求调用HandlerMapping(处理映射器)
- HandlerMapping找到具体的映射器,生成映射器对象及处理器拦截器(如果有),在一起返回给DispatchServlet
- DispatchServlet调用HandlerAdapter(处理器适配器)
- HandlerAdapter经过适配器调用具体的处理器(Handler/Controller)
- 方法上添加@ResponseBody
- 通过HttpMessageConverter来返回结果转化为JSON并响应。