Spring框架相关面试题

一、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工作流程原理?

  1. 用户发出请求到前端控制器DispatcherServlet
  2. DispatcherServlet收到请求调用HandlerMapping(处理映射器)
  3. HandlerMapping找到具体的映射器,生成映射器对象及处理器拦截器(如果有),在一起返回给DispatchServlet
  4. DispatchServlet调用HandlerAdapter(处理器适配器)
  5. HandlerAdapter经过适配器调用具体的处理器(Handler/Controller)
  6. 方法上添加@ResponseBody
  7. 通过HttpMessageConverter来返回结果转化为JSON并响应。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值