Spring八股 常见面试题

本文介绍了Spring框架中的Bean概念、注解如@Component、@Autowired和@Resource的区别,单例模式的线程安全性,AOP(面向切面编程)的应用,SpringMVC的核心组件,以及SpringBoot自动配置原理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Spring包含哪些部分

  1. Spring Core Container 核心容器模块。提供Spring框架的基本功能,如IoC(控制反转)和DI(依赖注入)
  2. Spring Context 上下文模块。提供Spring框架的基本功能,如IoC(控制反转)和DI(依赖注入)
  3. Spring AOP 面向切面编程模块
  4. Spring JDBC 数据访问模块。 封装了JDBC,简化了数据访问操作
  5. Spring ORM 对象关系映射模块。使用面向对象的方式操作数据库,提高了开发效率和代码的可维护性
  6. Spring Transaction 事务管理模块。提供了声明式事务管理和编程式事务管理两种方式。
  7. Spring Web Web 开发模块。提供了对Servlet、JSP、WebSocket等的支持。
  8. Spring MVC MVC 框架模块。实现了MVC设计模式,将Web应用划分为模型、视图和控制器三个层次,提高了代码的可读性和可维护性。
  9. Spring Security 安全模块。提供了各种安全功能,如认证、授权、加密等。
  10. Spring Test 测试模块。提供了对Spring应用程序的单元测试和集成测试支持。

什么是Spring Bean

简单来说,Bean 代指的就是那些被 IoC 容器所管理的对象。我们需要告诉 IoC 容器帮助我们管理哪些对象,这个是通过配置元数据来定义的。配置元数据可以是 XML 文件、注解或者 Java 配置类。

什么是IOC

IoC,即反转控制,其核心思路在于将创建和管理对象交给第三方进行,降低了对象之间的耦合度,使资源变得容易管理。

在Spring中,其最常见的实现模式是依赖注入(DI)

将一个类声明为 Bean 的注解有哪些?

  • @Component:通用的注解,可标注任意类为 Spring 组件。如果一个 Bean 不知道属于哪个层,可以使用@Component 注解标注。
  • @Repository : 对应持久层即 Dao 层,主要用于数据库相关操作。
  • @Service : 对应服务层,主要涉及一些复杂的逻辑,需要用到 Dao 层。
  • @Controller : 对应 Spring MVC 控制层,主要用于接受用户请求并调用 Service 层返回数据给前端页面。

注入Bean的注解有哪些

Spring 内置的 @Autowired 以及 JDK 内置的 @Resource@Inject 都可以用于注入 Bean。一般使用@Autowired@Resource

@Autowired@Resource区别

  • @Autowired 是 Spring 提供的注解,@Resource 是 JDK 提供的注解。
  • @Autowired默认注入方式是byType(根据类型匹配),@Resource 默认注入方式为byName(根据名称进行匹配)
  • 当一个接口存在多个实现类的情况下,@Autowired@Resource都需要通过名称才能正确匹配到对应的 Bean。Autowired 可以通过 @Qualifier 注解来显式指定名称,@Resource可以通过 name 属性来显式指定名称。
  • @Autowired 支持在构造函数、方法、字段和参数上使用。@Resource 主要用于字段和方法上的注入,不支持在构造函数或参数上使用。

Spring框架中的单例bean是线程安全的吗?

并非是线程安全的。

当多个用户请求同一个服务时,容器会给每一个请求分配一个线程。这时多个线程会并发执行该请求对应的成员方法。如果该处理逻辑中有对该例状态的修改,则必须考虑线程同步问题。由于Spring框架并没有对bean进行任何多线程的封装处理,关于单例bean的线程安全和并发问题需要开发者自行解决。

通常在项目中使用的spring bean都是不可变状态,所以在某种程度上说,Spring的单例bean是线程安全的。

但是如果bean有多种状态的话,可以通过加锁或者将bean改为多例解决,即将注释“@singleton”更改为“@prototype"。

手写单例

单例实现方式主要有三种

饿汉式

public class Singleton {
    private static Singleton uniqueInstance;
    private Singleton(){}

    public static Singleton getUniqueInstance() {
        if (uniqueInstance == null){
            synchronized (Singleton.class){
                if (uniqueInstance == null){
                    uniqueInstance = new Singleton();
                }
            }
        }
        return uniqueInstance;
    }
}
  • 先不创建实例,当第一次被调用时,再创建实例,所以被称为懒汉式。
  • 延迟了实例化,如果不需要使用该类,就不会被实例化,节约了系统资源。

饿汉式

public class Singleton {
    private static volatile Singleton uniqueInstance = new Singleton();
    private Singleton(){}

    public static Singleton getUniqueInstance() {
        
        return uniqueInstance;
    }
}
  • 直接先实例化好实例 (饿死鬼一样,所以称为饿汉式),然后当需要使用的时候,直接调方法就可以使用了。
  • 优点: 提前实例化好了一个实例,避免了线程不安全问题的出现。
  • 缺点: 直接实例化好了实例,不再延迟实例化;若系统没有使用这个实例,或者系统运行很久之后才需要使用这个实例,都会操作系统的资源浪费。

双重检查锁实现(线程安全)DCL

public class Singleton {
    private static volatile Singleton uniqueInstance;
    private Singleton(){}

    public static Singleton getUniqueInstance() {
        if (uniqueInstance == null){
            synchronized (Singleton.class){
                if (uniqueInstance == null){
                    uniqueInstance = new Singleton();
                }
            }
        }
        return uniqueInstance;
    }
}

Spring的bean的生命周期

  1. 通过BeanDefinition获取bean的定义信息
  2. 调用构造函数实例化bean
  3. bean的依赖注入
  4. 处理Aware接口(BeanNameAware、BeanFactoryAware、ApplicationContextAware)
  5. 执行BeanPostProcessor的前置初始化方法
  6. 初始化方法(InitializingBean、init-method)
    7.执行BeanPostProcessor的后置初始化方法
    在这里插入图片描述
BeanDefinition

Spring容器在进行实例化时,会将xml配置的的信息封装成一个BeanDefinition对象,Spring根据BeanDefinition来创建Bean对象,里面有很多的属性来表述Bean。

  • beanClassName: bean的类名
  • initMethodName: 初始化方法名称
  • properryValues: bean 的属性值
  • scope: 作用域
  • lazyInit: 延迟初始化

什么是AOP,你们项目中是否使用到AOP?

AOP称为面向切面编程,用于将与业务无关,但是对多个对象产生影响的公共行为和逻辑进行抽取和封装,形成一个可重用的模块,这个模块被命名为"切面"(Aspect)。可以减少系统中的重复代码,降低模块间的耦合度,同时提高了系统的可维护性。

术语含义
目标 Target被通知的对象
代理 Proxy向目标对象应用通知之后创建的代理对象
连接点 JoinPoint目标对象的所属类,定义的所有方法均为连接点
切入点 Pointcut被切面拦截/增强的连接点
通知 Advice拦截到目标对象的连接点后要做的事情
切面切入点+通知
织入 Weaving将通知应用到目标对象,进而生成代理对象的过程操作

常见AOP使用场景:

  • 记录操作日志
  • 缓存处理
  • Spring中内置的事务处理

项目中有没有使用到AOP

记录操作日志

核心是使用AOP中的环绕通知+切点表达式(找到要记录日志的方法),通过环绕通知的参数获取请求方法的参数(类、方法、注解、请求方式等),获取这些参数后,保存到数据库。

Spring中的事务如何实现

其本质是通过AOP功能,对方法前后进行拦截,在执行方法之前开启事务,在执行完目标方法后,根据执行情况提交或回滚事务。

Spring中事务失效的场景

  1. 异常捕获处理
  • 原因:事务通知只有捕捉到了目标抛出的异常,才能进行后续的回滚处理,如果目标自己处理掉异常,事务通知无法知悉。
  • 解决:在catch块中添加throw new RuntimeException(e) 抛出
  1. 抛出检查异常
  • 原因:Spring只会默认回滚非检查异常
  • 解决:配置rollbackFor属性,使得Spring回滚所有抛出异常
@Transactional(rollbackFor=Exception.class)
  1. 非public方法导致事务失效
  • 原因:Spring为方法创建代理、添加事务通知、前提条件都是该方法是public的
  • 解决:将方法改为public

Spring中的循环引用

  • 循环依赖:循环依赖即循环引用,也就是两个或以上的bean互相持有对方,最终形成闭环。比如A依赖于B,B依赖于A
  • 循环依赖在spring中是允许存在的,spring框架根据三级缓存已经解决了大部分的循环依赖
  1. 一级缓存:单例池,缓存已经经历了完整的生命周期,已经初始化完成的Bean对象
  2. 二级缓存:缓存早期的bean对象(生命周期还没走完)
  3. 三级缓存:缓存的是ObjectFactory,表示对象工厂,用于创建某个对象的代理对象。产生的代理对象同意存入二级缓存,需要使用时再取出
    在这里插入图片描述

构造方法出现循环依赖怎么解决

循环依赖的注入方式是构造函数
原因: 由于bean对象的生命周期中构造函数是第一个执行的,spring框架并不能解决构造函数的依赖注入
解决方法:使用@lazy进行懒加载,什么时候需要对象再进行bean对象的创建

public A(@Lazy B){
    System.out.println("A的构造方法执行完成");
    this.b = b;
        
}

SpringMVC是什么

MVC 是模型(Model)、视图(View)、控制器(Controller)的简写,其核心思想是通过将业务逻辑、数据、显示分离来组织代码。

Spring MVC 的核心组件有哪些?

  • DispatcherServlet核心的中央处理器,负责接收请求、分发,并给予客户端响应。
  • HandlerMapping处理器映射器,根据 URL 去匹配查找能处理的 Handler ,并会将请求涉及到的拦截器和 Handler 一起封装。
  • HandlerAdapter:处理器适配器,根据 HandlerMapping 找到的 Handler ,适配执行对应的 Handler
  • Handler:请求处理器,处理实际请求的处理器。
  • ViewResolver:视图解析器,根据 Handler 返回的逻辑视图 / 视图,解析并渲染真正的视图,并传递给 DispatcherServlet 响应客户端

SpringMVC的执行流程

  1. 用户发出请求到中央处理器DispatcherServlet
  2. DispatcherServlet收到请求调用处理器映射器(HandlerMapping)
  3. HandlerMapping找到jurisdiction的处理器,生成处理器对象及处理器拦截器(如果存在),再一起返回给DispatcherServlet
  4. DispatcherServlet调用处理器适配器(HandlerAdapter)
  5. HandlerAdapter经过适配器调用具体的处理器(Handler/Controller)
  6. 方法上添加@ResponseBody
  7. 通过HttpMessageConverter来返回结果转换为json并响应。
    在这里插入图片描述

Springboot自动配置原理

  1. 在Spring Boot项目中的引导类上有一个注解@SpringBootApplication,这个注解是对三个注解进行封装,分别是:
  • @SpringBootConfiguration
  • @EnableAutoConfiguration
  • @ComponentScan
  1. 其中@EnableAutoConfiguration是实现自动化配置的核心注解。该注解通过@Import注解导入对应的配置选择器。内部就是读取了该项目和该项目引用的Jar包的classpath路径下META-INF/spring.factories文件中的所有配置的类的全类名。在这些配置类中所定义的Bean会根据条件注释所指定的条件来决定是否需要将其导入到Spring容器中。
  2. 条件判断会有@ConditionalOnClass这样的注解,判断是否有对应的class文件。如果有则加载该类,把这个配置类的所有Bean放入spring容器中使用。

SpringMVC常见的注解

在这里插入图片描述

SpringBoot常见的注解

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值