2021面试准备-Spring 框架

118 阅读18分钟

五.Spring 框架

5.1 Spring 是什么?

Spring是一个轻量级的IoC和AOP容器框架。是为Java应用程序提供基 础性服务的一套框架,目的是用于简化企业应用程序的开发,它使得开发者只需 要关心业务需求。常见的配置方式有三种:基于XML的配置.基于注解的配置. 基于Java的配置。

​ 主要由以下几个模块组成: Spring Core:核心类库,提供IOC服务;

​ Spring Context:提供框架式的Bean访问方式,以及企业级功能(JNDI. 定时任务等);

​ Spring AOP:AOP服务; Spring DAO:对JDBC的抽象,简化了数据访问异常的处理;

​ Spring ORM:对现有的ORM框架的支持;

​ Spring Web:提供了基本的面向Web的综合特性,例如多方文件上传;

​ Spring MVC:提供面向Web应用的Model-View-Controller实现。

5.2 Spring的AOP理解?

​ OOP面向对象,允许开发者定义纵向的关系,但并适用于定义横向的关系, 导致了大量代码的重复,而不利于各个模块的重用

​ AOP,一般称为面向切面,作为面向对象的一种补充,用于将那些与业务 无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的 模块,这个模块被命名为“切面”(Aspect),减少系统中的重复代码,降低 了模块间的耦合度,同时提高了系统的可维护性。可用于权限认证.日志.事务处理。

​ AOP实现的关键在于代理模式,AOP代理主要分为静态代理和动态代理。 静态代理的代表为AspectJ;动态代理则以Spring AOP为代表。

​ (1)AspectJ是静态代理的增强,所谓静态代理,就是AOP框架会在编 译阶段生成AOP代理类,因此也称为编译时增强,他会在编译阶段将AspectJ(切 面)织入到Java字节码中,运行的时候就是增强之后的AOP对象。

​ (2)Spring AOP使用的动态代理,所谓的动态代理就是说AOP框架不会 去修改字节码,而是每次运行时在内存中临时为方法生成一个AOP对象,这个 AOP对象包含了目标对象的全部方法,并且在特定的切点做了增强处理,并回 调原对象的方法。

​ Spring AOP中的动态代理主要有两种方式,JDK动态代理和CGLIB动态 代理:

​ (1)JDK动态代理只提供接口代理,不支持类代理,核心InvocationHandler 接口和Proxy类,InvocationHandler 通过invoke()方法反射来调用目标类中 的代码,动态地将横切逻辑和业务编织在一起,Proxy利用 InvocationHandler 动态创建一个符合某一接口的的实例, 生成目标类的代理对象。

​ (2)如果代理类没有实现 InvocationHandler 接口,那么Spring AOP会 选择使用CGLIB来动态代理目标类。CGLIB(Code Generation Library),是 一个代码生成的类库,可以在运行时动态的生成指定类的一个子类对象,并覆盖 其中特定方法并添加增强代码,从而实现AOP。CGLIB是通过继承的方式做的动态代理,因此如果某个类被标记为final,那么它是无法使用CGLIB做动态代理的

​ (3)静态代理与动态代理区别在于生成AOP代理对象的时机不同,相对 来说AspectJ的静态代理方式具有更好的性能,但是AspectJ需要特定的编译 器进行处理,而Spring AOP则无需特定的编译器处理

InvocationHandler invoke(Object  proxy,Method  method,Object[] args):proxy是最终生成的代理实例;  method 是被代理目标实例的某个具体 方法;args 是被代理目标实例某个方法的具体入参, 在方法反射调用时使用。 

5.3 Spring的IOC理解?

​ (1)IOC就是控制反转,是指创建对象的控制权的转移,以前创建对象的 主动权和时机是由自己把控的,而现在这种权力转移到Spring容器中,并由容 器根据配置文件去创建实例和管理各个实例之间的依赖关系,对象与对象之间松 散耦合,也利于功能的复用。DI依赖注入,和控制反转是同一个概念的不同角 度的描述,即 应用程序在运行时依赖IoC容器来动态注入对象需要的外部资源

​ (2)最直观的表达就是,IOC让对象的创建不用去new了,可以由spring 自动生产,使用java的反射机制,根据配置文件在运行时动态的去创建对象以 及管理对象,并调用对象的方法的。

​ (3)Spring的IOC有三种注入方式 :构造器注入, setter方法注入, 根据 注解注入

​ IOC让相互协作的组件保持松散的耦合,而AOP编程允许你把遍布于应用 各层的功能分离出来形成可重用的功能组件。

5.4 BeanFactory和ApplicationContext有什么区别?

​ BeanFactory和ApplicationContext是Spring的两大核心接口,都可以 当做Spring的容器。其中ApplicationContext是BeanFactory的子接口。

​ (1)BeanFactory:是 Spring里面最底层的接口,包含了各种Bean的定 义,读取bean配置文档,管理bean的加载.实例化,控制bean的生命周期, 维护bean之间的依赖关系。ApplicationContext接口作为BeanFactory的派 生。

​ (2)BeanFactroy采用的是延迟加载形式来注入Bean的,即只有在使用 到某个Bean时(调用getBean()),才对该Bean进行加载实例化。这样,我们 就不能发现一些存在的Spring的配置问题。如果Bean的某一个属性没有注入, BeanFacotry加载后,直至第一次使用调用getBean方法才会抛出异常。 ApplicationContext,它是在容器启动时,一次性创建了所有的Bean。这 样,在容器启动时,我们就可以发现Spring中存在的配置错误,这样有利于检查所依赖属性是否注入。 ApplicationContext启动后预载入所有的单实例 Bean,通过预载入单实例bean ,确保当你需要的时候,你就不用等待,因为它 们已经创建好了。 相对于基本的BeanFactory,ApplicationContext 唯一的不足是占用内存 空间。当应用程序配置Bean较多时,程序启动较慢。

​ (3)BeanFactory通常以编程的方式被创建,ApplicationContext还能 以声明的方式创建,如使用ContextLoader。

​ (4)BeanFactory和ApplicationContext都支持。 BeanPostProcessor.BeanFactoryPostProcessor的使用,但两者之间的区别 是:BeanFactory需要手动注册,而ApplicationContext则是自动注册。

5.5 请解释Spring Bean的生命周期?

​ 首先说一下Servlet的生命周期:实例化,初始init,接收请求service, 销毁destroy;

​ Spring上下文中的Bean生命周期也类似,如下:

​ (1)实例化Bean:

​ 对于BeanFactory容器,当客户向容器请求一个尚未初始化的bean时, 或初始化bean的时候需要注入另一个尚未初始化的依赖时,容器就会调用 createBean进行实例化。对于ApplicationContext容器,当容器启动结束后, 通过获取BeanDefinition 对象中的信息,实例化所有的bean。

(2)设置对象属性(依赖注入): 实例化后的对象被封装在BeanWrapper对象中,紧接着,Spring根据 BeanDefinition中的信息 以及 通过BeanWrapper提供的设置属性的接口完 成依赖注入。 (3)处理Aware接口: 接着,Spring会检测该对象是否实现了xxxAware接口,并将相关的 xxxAware实例注入给Bean: 如果这个Bean已经实现了BeanNameAware接口,会调用它实现的 setBeanName(String beanId)方法,此处传递的就是Spring配置文件中Bean 的id值; 如果这个Bean已经实现了BeanFactoryAware接口,会调用它实现的 setBeanFactory()方法,传递的是Spring工厂自身。

如果这个Bean已经实现了ApplicationContextAware接口,会调用 setApplicationContext(ApplicationContext)方法,传入Spring上下文;

(4)BeanPostProcessor: 如果想对Bean进行一些自定义的处理,那么可以让Bean实现了 BeanPostProcessor接口,那将会调用 postProcessBeforeInitialization(Object obj, String s)方法。由于这个方法是 在Bean初始化结束时调用的,所以可以被应用于内存或缓存技术;

(5)InitializingBean 与 init-method: 如果Bean在Spring配置文件中配置了 init-method 属性,则会自动调 用其配置的初始化方法。

(6)如果这个Bean实现了BeanPostProcessor接口,将会调用 postProcessAfterInitialization(Object obj, String s)方法; 以上几个步骤完成后,Bean就已经被正确创建了,之后就可以使用这个 Bean了。

(7)DisposableBean: 当Bean不再需要时,会经过清理阶段,如果Bean实现了DisposableBean 这个接口,会调用其实现的destroy()方法;

(8)destroy-method: 最后,如果这个Bean的Spring配置中配置了destroy-method属性,会 自动调用其配置的销毁方法img

在这里插入图片描述

5.6 解释Spring支持的几种bean的作用域。

​ Spring容器中的bean可以分为5个范围:

​ (1)singleton:默认,每个容器中只有一个bean的实例,单例的模式由 BeanFactory自身来维护。

​ (2)prototype:为每一个bean请求提供一个实例。

​ (3)request:为每一个网络请求创建一个实例,在请求完成以后,bean 会失效并被垃圾回收器回收。

​ (4)session:与request范围类似,确保每个session中有一个bean 的实例,在session过期后,bean会随之失效。

​ (5)global-session:全局作用域,global-session和Portlet应用相关。 当你的应用部署在Portlet容器中工作时,它包含很多portlet。如果你想要声

明让所有的portlet共用全局的存储变量的话,那么这全局变量需要存储在 global-session中。全局作用域与Servlet中的session作用域效果相同。

5.7 spring常见的注解有哪些?

@Configuration把一个类作为一个IoC容器,它的某个方法头上如果注册了

@Bean,就会作为这个Spring容器中的Bean。

@Scope注解 作用域 @Lazy(true) 表示延迟初始化 @Service用于标注业务层组件.

@Controller用于标注控制层组件(如struts中的action)

@Repository用于标注数据访问组件,即DAO组件。

@Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行 标注。

@Scope用于指定scope作用域的(用在类上)

@PostConstruct用于指定初始化方法(用在方法上)

@PreDestory用于指定销毁方法(用在方法上)

@Resource 默认按名称装配,当找不到与名称匹配的bean才会按类型装配。

@DependsOn:定义Bean初始化及销毁时的顺序

@Primary:自动装配时当出现多个Bean候选者时,被注解为@Primary的 Bean将作为首选者,否则将抛出异常

@Resource 默认按名称装配,当找不到与名称匹配的bean才会按类型装配。

@Autowired 默认按类型装配,如果我们想使用按名称装配,可以结合 @Qualifier注解一起使用。如下:@Autowired@Qualifier("personDaoBean") 存在多个实例配合使用

5.8 Spring框架中的单例Beans是线程安全的么?

​ Spring框架并没有对单例bean进行任何多线程的封装处理。关于单例 bean的线程安全和并发问题需要开发者自行去搞定。但实际上,大部分的 Spring bean并没有可变的状态(比如Serview类和DAO类),所以在某种程度 上说Spring的单例bean是线程安全的。如果你的bean有多种状态的话(比 如 View Model 对象),就需要自行保证线程安全。最浅显的解决办法就是将 多态bean的作用域由“singleton”变更为“prototype”。

5.9 Spring如何处理线程并发问题?

​ 在一般情况下,只有无状态的Bean才可以在多线程环境下共享,在Spring 中,绝大部分Bean都可以声明为singleton作用域,因为Spring对一些Bean 中非线程安全状态采用ThreadLocal进行处理,解决线程安全问题。

	ThreadLocal和线程同步机制都是为了解决多线程中相同变量的访问冲突 问题。同步机制采用了“时间换空间”的方式,仅提供一份变量,不同的线程在 访问前需要获取锁,没获得锁的线程则需要排队。而ThreadLocal采用了“空 间换时间”的方式。 ThreadLocal会为每一个线程提供一个独立的变量副本,从而隔离了多个线 程对数据的访问冲突。因为每一个线程都拥有自己的变量副本,从而也就没有必 要对该变量进行同步了。

​ ThreadLocal提供了线程安全的共享对象,在编写多线 程代码时,可以把不安全的变量封装进ThreadLocal。

5.10 Spring基于xml注入bean的几种方式?

(1)Set方法注入;

(2)构造器注入:

​ 1.通过index设置参数的位置;

​ 2.通过type设置参数类型;

​ 3.通过name注入;

(3)静态工厂注入;

(4)实例工厂;

5.11 Spring 框架中都用到了哪些设计模式?

(1)工厂模式:BeanFactory就是简单工厂模式的体现,用来创建对象的 实例;

(2)单例模式:Bean默认为单例模式。

(3)代理模式:Spring的AOP功能用到了JDK的动态代理和CGLIB字节 码生成技术;

(4)模板方法:用来解决代码重复的问题。 比如. RestTemplate, JmsTemplate, JpaTemplate。

(5)观察者模式:定义对象键一种一对多的依赖关系,当一个对象的状态 发生改变时,所有依赖于它的对象都会得到通知被制动更新,如Spring中 listener的实现--ApplicationListener。

5.12 Spring事务的实现方式和实现原理?

​ Spring事务的本质其实就是数据库对事务的支持,没有数据库的事务支持, spring是无法提供事务功能的。真正的数据库层的事务提交和回滚是通过 binlog或者redo log实现的。

(1)Spring事务的种类:

​ spring支持编程式事务管理和声明式事务管理两种方式:

​ A.编程式事务管理使用 TransactionTemplate。

​ B.声明式事务管理建立在AOP之上的。其本质是通过AOP功能,对方法前 后进行拦截,将事务处理的功能编织到拦截的方法中,也就是在目标方法开始之 前加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。

​ 声明式事务最大的优点就是不需要在业务逻辑代码中掺杂事务管理的代码, 只需在配置文件中做相关的事务规则声明或通过@Transactional注解的方式, 便可以将事务规则应用到业务逻辑中。

​ 声明式事务管理要优于编程式事务管理,这正是spring倡导的非侵入式的 开发方式,使业务代码不受污染,只要加上注解就可以获得完全的事务支持。唯 一不足地方是,最细粒度只能作用到方法级别,无法做到像编程式事务那样可以 作用到代码块级别。

(2)Spring的事务传播行为:

​ Spring事务的传播行为说的是,当多个事务同时存在的时候,Spring如何 处理这些事务的行为。

​ ① PROPAGATION_REQUIRED:如果当前没有事务,就创建一个新事务, 如果当前存在事务,就加入该事务,该设置是最常用的设置。

​ ② PROPAGATION_SUPPORTS:支持当前事务,如果当前存在事务,就 加入该事务,如果当前不存在事务,就以非事务执行。

​ ‘③ PROPAGATION_MANDATORY:支持当前事务,如果当前存在事务, 就加入该事务,如果当前不存在事务,就抛出异常。

​ ④ PROPAGATION_REQUIRES_NEW:创建新事务,无论当前存不存在事 务,都创建新事务。

​ ⑤ PROPAGATION_NOT_SUPPORTED:以非事务方式执行操作,如果当 前存在事务,就把当前事务挂起。

​ ⑥ PROPAGATION_NEVER:以非事务方式执行,如果当前存在事务,则 抛出异常。

​ ⑦ PROPAGATION_NESTED:如果当前存在事务,则在嵌套事务内执行。 如果当前没有事务,则按REQUIRED属性执行。

(3)Spring中的隔离级别

​ ① ISOLATION_DEFAULT:这是个 PlatfromTransactionManager 默认的 隔离级别,使用数据库默认的事务隔离级别。

​ ② ISOLATION_READ_UNCOMMITTED:读未提交,允许另外一个事务可 以看到这个事务未提交的数据。

​ ③ ISOLATION_READ_COMMITTED:读已提交,保证一个事务修改的数 据提交后才能被另一事务读取,而且能看到该事务对已有记录的更新。

​ ④ ISOLATION_REPEATABLE_READ:可重复读,保证一个事务修改的数据 提交后才能被另一事务读取,但是不能看到该事务对已有记录的更新。

​ ⑤ ISOLATION_SERIALIZABLE:一个事务在执行的过程中完全看不到其他 事务对数据库所做的更新。

5.13 解释一下Spring AOP里面的几个名词?

(1)切面(Aspect):被抽取的公共模块,可能会横切多个对象。 在Spring AOP中,切面可以使用通用类(基于模式的风格) 或者在普通类中 以 @AspectJ 注解来实现。

(2)连接点(Join point):指方法,在Spring AOP中,一个连接点 总 是 代表一个方法的执行。

(3)通知(Advice):在切面的某个特定的连接点(Join point)上执行 的动作。通知有各种类型,其中包括“around”.“before”和“after”等通知。 许多AOP框架,包括Spring,都是以拦截器做通知模型, 并维护一个以连接 点为中心的拦截器链

(4)切入点(Pointcut):切入点是指 我们要对哪些Join point进行拦截 的定义。通过切入点表达式,指定拦截的方法,比如指定拦截add*.search*。

(5)引入(Introduction):(也被称为内部类型声明(inter-type declaration))。声明额外的方法或者某个类型的字段。Spring允许引入新的 接口(以及一个对应的实现)到任何被代理的对象。例如,你可以使用一个引入 来使bean实现 IsModified 接口,以便简化缓存机制。

(6)目标对象(Target Object): 被一个或者多个切面(aspect)所通 知(advise)的对象。也有人把它叫做 被通知(adviced) 对象。 既然Spring AOP是通过运行时代理实现的,这个对象永远是一个 被代理(proxied) 对象。

(7)织入(Weaving):指把增强应用到目标对象来创建新的代理对象的 过程。Spring是在运行时完成织入。

切入点(pointcut)和连接点(join point)匹配的概念是AOP的关键,这 使得AOP不同于其它仅仅提供拦截功能的旧技术。 切入点使得定位通知 (advice)可独立于OO层次。 例如,一个提供声明式事务管理的around通 知可以被应用到一组横跨多个对象中的方法上(例如服务层的所有业务操作)。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ROQd8xvB-1610441296451)(C:\Users\Administrator\Desktop\面试\media\image-20210112164648994.png)]

5.14 Spring通知有哪些类型?

(1)前置通知(Before advice):在某连接点(join point)之前执行的 通知,但这个通知不能阻止连接点前的执行(除非它抛出一个异常)。

(2)返回后通知(After returning advice):在某连接点(join point) 正常完成后执行的通知:例如,一个方法没有抛出任何异常,正常返回。

(3)抛出异常后通知(After throwing advice):在方法抛出异常退出时 执行的通知。

(4)后通知(After (finally) advice):当某连接点退出的时候执行的通知 (不论是正常返回还是异常退出)。

前执行的 通知,但这个通知不能阻止连接点前的执行(除非它抛出一个异常)。

(2)返回后通知(After returning advice):在某连接点(join point) 正常完成后执行的通知:例如,一个方法没有抛出任何异常,正常返回。

(3)抛出异常后通知(After throwing advice):在方法抛出异常退出时 执行的通知。

(4)后通知(After (finally) advice):当某连接点退出的时候执行的通知 (不论是正常返回还是异常退出)。

(5)环绕通知(Around Advice):包围一个连接点(join point)的通知, 如方法调用。这是最强大的一种通知类型。 环绕通知可以在方法调用前后完成 自定义的行为。它也会选择是否继续执行连接点或直接返回它们自己的返回值或 抛出异常来结束执行。 环绕通知是最常用的一种通知类型。大部分基于拦截的 AOP框架,例如Nanning和JBoss4,都只提供环绕通知。