49个Spring经典面试题总结,附带答案,赶紧收藏
1. 一般问题
1.1. 不同版本的 spring framework 有哪些主要功能?
version | feature |
---|---|
spring 2.5 | 发布于 2007 年。这是第一个支持注解的版本。 |
spring 3.0 | 发布于 2009 年。它完全利用了 java5 中的改进,并为 jee6 提供了支持。 |
spring 4.0 | 发布于 2013 年。这是第一个完全支持 java8 的版本。 |
1.2. 什么是 spring framework?
- spring 是一个开源应用框架,旨在降低应用程序开发的复杂度。
- 它是轻量级、松散耦合的。
- 它具有分层体系结构,允许用户选择组件,同时还为 j2ee 应用程序开发提供了一个有凝聚力的框架。
- 它可以集成其他框架,如 structs、hibernate、ejb 等,所以又称为框架的框架。
1.3. 列举 spring framework 的优点。
- 由于 spring frameworks 的分层架构,用户可以*选择自己需要的组件。
- spring framework 支持 pojo(plain old java object) 编程,从而具备持续集成和可测试性。
- 由于依赖注入和控制反转,jdbc 得以简化。
- 它是开源免费的。
1.4. spring framework 有哪些不同的功能?
- 轻量级 - spring 在代码量和透明度方面都很轻便。
- ioc - 控制反转
- aop - 面向切面编程可以将应用业务逻辑和系统服务分离,以实现高内聚。
- 容器 - spring 负责创建和管理对象(bean)的生命周期和配置。
- mvc - 对 web 应用提供了高度可配置性,其他框架的集成也十分方便。
- 事务管理 - 提供了用于事务管理的通用抽象层。spring 的事务支持也可用于容器较少的环境。
- jdbc 异常 - spring 的 jdbc 抽象层提供了一个异常层次结构,简化了错误处理策略。
1.5. spring framework 中有多少个模块,它们分别是什么?
-
spring 核心容器 – 该层基本上是 spring framework 的核心。它包含以下模块:
- spring core
- spring bean
- spel (spring expression language)
- spring context
-
数据访问/集成 – 该层提供与数据库交互的支持。它包含以下模块:
- jdbc (java database connectivity)
- orm (object relational mapping)
- oxm (object xml mappers)
- jms (java messaging service)
- transaction
-
web – 该层提供了创建 web 应用程序的支持。它包含以下模块:aop – 该层支持面向切面编程
- web
- web – servlet
- web – socket
- web – portlet
- instrumentation – 该层为类检测和类加载器实现提供支持。
- test – 该层为使用 junit 和 testng 进行测试提供支持。
-
几个杂项模块:
- messaging – 该模块为 stomp 提供支持。它还支持注解编程模型,该模型用于从 websocket 客户端路由和处理 stomp 消息。
- aspects – 该模块为与 aspectj 的集成提供支持。
1.6. 什么是 spring 配置文件?
spring 配置文件是 xml 文件。该文件主要包含类信息。它描述了这些类是如何配置以及相互引入的。但是,xml 配置文件冗长且更加干净。如果没有正确规划和编写,那么在大项目中管理变得非常困难。
1.7. spring 应用程序有哪些不同组件?
spring 应用一般有以下组件:
- 接口 - 定义功能。
- bean 类 - 它包含属性,setter 和 getter 方法,函数等。
- spring 面向切面编程(aop) - 提供面向切面编程的功能。
- bean 配置文件 - 包含类的信息以及如何配置它们。
- 用户程序 - 它使用接口。
1.8. 使用 spring 有哪些方式?
使用 spring 有以下方式:
- 作为一个成熟的 spring web 应用程序。
- 作为第三方 web 框架,使用 spring frameworks 中间层。
- 用于远程使用。
- 作为企业级 java bean,它可以包装现有的 pojo(plain old java objects)。
2. 依赖注入(ioc)
2.1. 什么是 spring ioc 容器?
spring 框架的核心是 spring 容器。容器创建对象,将它们装配在一起,配置它们并管理它们的完整生命周期。spring 容器使用依赖注入来管理组成应用程序的组件。容器通过读取提供的配置元数据来接收对象进行实例化,配置和组装的指令。该元数据可以通过 xml,java 注解或 java 代码提供。
2.2. 什么是依赖注入?
在依赖注入中,您不必创建对象,但必须描述如何创建它们。您不是直接在代码中将组件和服务连接在一起,而是描述配置文件中哪些组件需要哪些服务。由 ioc 容器将它们装配在一起。
2.3. 可以通过多少种方式完成依赖注入?
通常,依赖注入可以通过三种方式完成,即:
- 构造函数注入
- setter 注入
- 接口注入
在 spring framework 中,仅使用构造函数和 setter 注入。
2.4. 区分构造函数注入和 setter 注入。
构造函数注入 | setter 注入 |
---|---|
没有部分注入 | 有部分注入 |
不会覆盖 setter 属性 | 会覆盖 setter 属性 |
任意修改都会创建一个新实例 | 任意修改不会创建一个新实例 |
适用于设置很多属性 | 适用于设置少量属性 |
2.5. spring 中有多少种 ioc 容器?
- beanfactory - beanfactory 就像一个包含 bean 集合的工厂类。它会在客户端要求时实例化 bean。
- applicationcontext - applicationcontext 接口扩展了 beanfactory 接口。它在 beanfactory 基础上提供了一些额外的功能。
2.6. 区分 beanfactory 和 applicationcontext。
beanfactory | applicationcontext |
---|---|
它使用懒加载 | 它使用即时加载 |
它使用语法显式提供资源对象 | 它自己创建和管理资源对象 |
不支持国际化 | 支持国际化 |
不支持基于依赖的注解 | 支持基于依赖的注解 |
2.7. 列举 ioc 的一些好处。
ioc 的一些好处是:
- 它将最小化应用程序中的代码量。
- 它将使您的应用程序易于测试,因为它不需要单元测试用例中的任何单例或 jndi 查找机制。
- 它以最小的影响和最少的侵入机制促进松耦合。
- 它支持即时的实例化和延迟加载服务。
2.8. spring ioc 的实现机制。
spring 中的 ioc 的实现原理就是工厂模式加反射机制。
示例:
interface fruit { public abstract void eat(); } class apple implements fruit { public void eat(){ system.out.println("apple"); } } class orange implements fruit { public void eat(){ system.out.println("orange"); } } class factory { public static fruit getinstance(string classname) { fruit f=null; try { f=(fruit)class.forname(classname).newinstance(); } catch (exception e) { e.printstacktrace(); } return f; } } class client { public static void main(string[] a) { fruit f=factory.getinstance("io.github.dunwu.spring.apple"); if(f!=null){ f.eat(); } } }
3. beans
3.1. 什么是 spring bean?
- 它们是构成用户应用程序主干的对象。
- bean 由 spring ioc 容器管理。
- 它们由 spring ioc 容器实例化,配置,装配和管理。
- bean 是基于用户提供给容器的配置元数据创建。
3.2. spring 提供了哪些配置方式?
- 基于 xml 配置
bean 所需的依赖项和服务在 xml 格式的配置文件中指定。这些配置文件通常包含许多 bean 定义和特定于应用程序的配置选项。它们通常以 bean 标签开头。例如:
<bean id="studentbean" class="org.edureka.firstspring.studentbean"> <property name="name" value="edureka"></property> </bean>
- 基于注解配置
您可以通过在相关的类,方法或字段声明上使用注解,将 bean 配置为组件类本身,而不是使用 xml 来描述 bean 装配。默认情况下,spring 容器中未打开注解装配。因此,您需要在使用它之前在 spring 配置文件中启用它。例如:
<beans> <context:annotation-config/> <!-- bean definitions go here --> </beans>
- 基于 java api 配置
spring 的 java 配置是通过使用 @bean 和 @configuration 来实现。
例如:
@configuration public class studentconfig { @bean public studentbean mystudent() { return new studentbean(); } }
3.3. spring 支持集中 bean scope?
spring bean 支持 5 种 scope:
- singleton - 每个 spring ioc 容器仅有一个单实例。
- prototype - 每次请求都会产生一个新的实例。
- request - 每一次 http 请求都会产生一个新的实例,并且该 bean 仅在当前 http 请求内有效。
- session - 每一次 http 请求都会产生一个新的 bean,同时该 bean 仅在当前 http session 内有效。
- global-session - 类似于标准的 http session 作用域,不过它仅仅在基于 portlet 的 web 应用中才有意义。portlet 规范定义了全局 session 的概念,它被所有构成某个 portlet web 应用的各种不同的 portlet 所共享。在 global session 作用域中定义的 bean 被限定于全局 portlet session 的生命周期范围内。如果你在 web 中使用 global session 作用域来标识 bean,那么 web 会自动当成 session 类型来使用。
仅当用户使用支持 web 的 applicationcontext 时,最后三个才可用。
3.4. spring bean 容器的生命周期是什么样的?
spring bean 容器的生命周期流程如下:
- spring 容器根据配置中的 bean 定义中实例化 bean。
- spring 使用依赖注入填充所有属性,如 bean 中所定义的配置。
- 如果 bean 实现 beannameaware 接口,则工厂通过传递 bean 的 id 来调用 setbeanname()。
- 如果 bean 实现 beanfactoryaware 接口,工厂通过传递自身的实例来调用 setbeanfactory()。
- 如果存在与 bean 关联的任何 beanpostprocessors,则调用 preprocessbeforeinitialization() 方法。
- 如果为 bean 指定了 init 方法(
<bean>
的 init-method 属性),那么将调用它。 - 最后,如果存在与 bean 关联的任何 beanpostprocessors,则将调用 postprocessafterinitialization() 方法。
- 如果 bean 实现 disposablebean 接口,当 spring 容器关闭时,会调用 destory()。
- 如果为 bean 指定了 destroy 方法(
<bean>
的 destroy-method 属性),那么将调用它。
3.5. 什么是 spring 的内部 bean?
只有将 bean 用作另一个 bean 的属性时,才能将 bean 声明为内部 bean。为了定义 bean,spring 的基于 xml 的配置元数据在 <property>
或 <constructor-arg>
中提供了 <bean>
元素的使用。内部 bean 总是匿名的,它们总是作为原型。
例如,假设我们有一个 student 类,其中引用了 person 类。这里我们将只创建一个 person 类实例并在 student 中使用它。
student.java
public class student { private person person; //setters and getters } public class person { private string name; private string address; //setters and getters }
bean.xml
<bean id=“studentbean" class="com.edureka.student"> <property name="person"> <!--this is inner bean --> <bean class="com.edureka.person"> <property name="name" value=“scott"></property> <property name="address" value=“bangalore"></property> </bean> </property> </bean>
3.6. 什么是 spring 装配
当 bean 在 spring 容器中组合在一起时,它被称为装配或 bean 装配。 spring 容器需要知道需要什么 bean 以及容器应该如何使用依赖注入来将 bean 绑定在一起,同时装配 bean。
3.7. 自动装配有哪些方式?
spring 容器能够自动装配 bean。也就是说,可以通过检查 beanfactory 的内容让 spring 自动解析 bean 的协作者。
自动装配的不同模式:
- no - 这是默认设置,表示没有自动装配。应使用显式 bean 引用进行装配。
- byname - 它根据 bean 的名称注入对象依赖项。它匹配并装配其属性与 xml 文件中由相同名称定义的 bean。
- bytype - 它根据类型注入对象依赖项。如果属性的类型与 xml 文件中的一个 bean 名称匹配,则匹配并装配属性。
- 构造函数 - 它通过调用类的构造函数来注入依赖项。它有大量的参数。
- autodetect - 首先容器尝试通过构造函数使用 autowire 装配,如果不能,则尝试通过 bytype 自动装配。
3.8. 自动装配有什么局限?
- 覆盖的可能性 - 您始终可以使用
<constructor-arg>
和<property>
设置指定依赖项,这将覆盖自动装配。 - 基本元数据类型 - 简单属性(如原数据类型,字符串和类)无法自动装配。
- 令人困惑的性质 - 总是喜欢使用明确的装配,因为自动装配不太精确。
4. 注解
4.1. 你用过哪些重要的 spring 注解?
- @controller - 用于 spring mvc 项目中的控制器类。
- @service - 用于服务类。
- @requestmapping - 用于在控制器处理程序方法中配置 uri 映射。
- @responsebody - 用于发送 object 作为响应,通常用于发送 xml 或 json 数据作为响应。
- @pathvariable - 用于将动态值从 uri 映射到处理程序方法参数。
- @autowired - 用于在 spring bean 中自动装配依赖项。
- @qualifier - 使用 @autowired 注解,以避免在存在多个 bean 类型实例时出现混淆。
- @scope - 用于配置 spring bean 的范围。
- @configuration,@componentscan 和 @bean - 用于基于 java 的配置。
- @aspect,@before,@after,@around,@pointcut - 用于切面编程(aop)。
4.2. 如何在 spring 中启动注解装配?
默认情况下,spring 容器中未打开注解装配。因此,要使用基于注解装配,我们必须通过配置<context:annotation-config />
元素在 spring 配置文件中启用它。
4.3. @component, @controller, @repository, @service 有何区别?
- @component:这将 java 类标记为 bean。它是任何 spring 管理组件的通用构造型。spring 的组件扫描机制现在可以将其拾取并将其拉入应用程序环境中。
- @controller:这将一个类标记为 spring web mvc 控制器。标有它的 bean 会自动导入到 ioc 容器中。
- @service:此注解是组件注解的特化。它不会对 @component 注解提供任何其他行为。您可以在服务层类中使用 @service 而不是 @component,因为它以更好的方式指定了意图。
- @repository:这个注解是具有类似用途和功能的 @component 注解的特化。它为 dao 提供了额外的好处。它将 dao 导入 ioc 容器,并使未经检查的异常有资格转换为 spring dataaccessexception。
4.4. @required 注解有什么用?
@required 应用于 bean 属性 setter 方法。此注解仅指示必须在配置时使用 bean 定义中的显式属性值或使用自动装配填充受影响的 bean 属性。如果尚未填充受影响的 bean 属性,则容器将抛出 beaninitializationexception。
示例:
public class employee { private string name; @required public void setname(string name){ this.name=name; } public string getname(){ return name; } }
4.5. @autowired 注解有什么用?
@autowired 可以更准确地控制应该在何处以及如何进行自动装配。此注解用于在 setter 方法,构造函数,具有任意名称或多个参数的属性或方法上自动装配 bean。默认情况下,它是类型驱动的注入。
public class employee { private string name; @autowired public void setname(string name) { this.name=name; } public string getname(){ return name; } }
4.6. @qualifier 注解有什么用?
当您创建多个相同类型的 bean 并希望仅使用属性装配其中一个 bean 时,您可以使用@qualifier 注解和 @autowired 通过指定应该装配哪个确切的 bean 来消除歧义。
例如,这里我们分别有两个类,employee 和 empaccount。在 empaccount 中,使用@qualifier 指定了必须装配 id 为 emp1 的 bean。
employee.java
public class employee { private string name; @autowired public void setname(string name) { this.name=name; } public string getname() { return name; } }
empaccount.java
public class empaccount { private employee emp; @autowired @qualifier(emp1) public void showname() { system.out.println(“employee name : ”+emp.getname); } }
4.7. @requestmapping 注解有什么用?
@requestmapping 注解用于将特定 http 请求方法映射到将处理相应请求的控制器中的特定类/方法。此注解可应用于两个级别:
- 类级别:映射请求的 url
- 方法级别:映射 url 以及 http 请求方法
5. 数据访问
5.1. spring dao 有什么用?
spring dao 使得 jdbc,hibernate 或 jdo 这样的数据访问技术更容易以一种统一的方式工作。这使得用户容易在持久性技术之间切换。它还允许您在编写代码时,无需考虑捕获每种技术不同的异常。
5.2. 列举 spring dao 抛出的异常。
5.3. spring jdbc api 中存在哪些类?
- jdbctemplate
- simplejdbctemplate
- namedparameterjdbctemplate
- simplejdbcinsert
- simplejdbccall
5.4. 使用 spring 访问 hibernate 的方法有哪些?
我们可以通过两种方式使用 spring 访问 hibernate:
- 使用 hibernate 模板和回调进行控制反转
- 扩展 hibernatedaosupport 并应用 aop 拦截器节点
5.5. 列举 spring 支持的事务管理类型
spring 支持两种类型的事务管理:
- 程序化事务管理:在此过程中,在编程的帮助下管理事务。它为您提供极大的灵活性,但维护起来非常困难。
- 声明式事务管理:在此,事务管理与业务代码分离。仅使用注解或基于 xml 的配置来管理事务。
5.6. spring 支持哪些 orm 框架
- hibernate
- ibatis
- jpa
- jdo
- ojb
6. aop
6.1. 什么是 aop?
aop(aspect-oriented programming), 即 面向切面编程, 它与 oop( object-oriented programming, 面向对象编程) 相辅相成, 提供了与 oop 不同的抽象软件结构的视角. 在 oop 中, 我们以类(class)作为我们的基本单元, 而 aop 中的基本单元是 aspect(切面)
6.2. aop 中的 aspect、advice、pointcut、jointpoint 和 advice 参数分别是什么?
- aspect - aspect 是一个实现交叉问题的类,例如事务管理。方面可以是配置的普通类,然后在 spring bean 配置文件中配置,或者我们可以使用 spring aspectj 支持使用 @aspect 注解将类声明为 aspect。
- advice - advice 是针对特定 joinpoint 采取的操作。在编程方面,它们是在应用程序中达到具有匹配切入点的特定 joinpoint 时执行的方法。您可以将 advice 视为 spring 拦截器(interceptor)或 servlet 过滤器(filter)。
- advice arguments - 我们可以在 advice 方法中传递参数。我们可以在切入点中使用 args() 表达式来应用于与参数模式匹配的任何方法。如果我们使用它,那么我们需要在确定参数类型的 advice 方法中使用相同的名称。
- pointcut - pointcut 是与 joinpoint 匹配的正则表达式,用于确定是否需要执行 advice。 pointcut 使用与 joinpoint 匹配的不同类型的表达式。spring 框架使用 aspectj pointcut 表达式语言来确定将应用通知方法的 joinpoint。
- joinpoint - joinpoint 是应用程序中的特定点,例如方法执行,异常处理,更改对象变量值等。在 spring aop 中,joinpoint 始终是方法的执行器。
6.3. 什么是通知(advice)?
特定 joinpoint 处的 aspect 所采取的动作称为 advice。spring aop 使用一个 advice 作为拦截器,在 joinpoint “周围”维护一系列的拦截器。
6.4. 有哪些类型的通知(advice)?
- before - 这些类型的 advice 在 joinpoint 方法之前执行,并使用 @before 注解标记进行配置。
- after returning - 这些类型的 advice 在连接点方法正常执行后执行,并使用@afterreturning 注解标记进行配置。
- after throwing - 这些类型的 advice 仅在 joinpoint 方法通过抛出异常退出并使用 @afterthrowing 注解标记配置时执行。
- after (finally) - 这些类型的 advice 在连接点方法之后执行,无论方法退出是正常还是异常返回,并使用 @after 注解标记进行配置。
- around - 这些类型的 advice 在连接点之前和之后执行,并使用 @around 注解标记进行配置。
6.5. 指出在 spring aop 中 concern 和 cross-cutting concern 的不同之处。
concern 是我们想要在应用程序的特定模块中定义的行为。它可以定义为我们想要实现的功能。
cross-cutting concern 是一个适用于整个应用的行为,这会影响整个应用程序。例如,日志记录,安全性和数据传输是应用程序几乎每个模块都需要关注的问题,因此它们是跨领域的问题。
6.6. aop 有哪些实现方式?
实现 aop 的技术,主要分为两大类:
-
静态代理 - 指使用 aop 框架提供的命令进行编译,从而在编译阶段就可生成 aop 代理类,因此也称为编译时增强;
- 编译时编织(特殊编译器实现)
- 类加载时编织(特殊的类加载器实现)。
-
动态代理 - 在运行时在内存中“临时”生成 aop 动态代理类,因此也被称为运行时增强。
- jdk 动态代理
- cglib
6.7. spring aop and aspectj aop 有什么区别?
spring aop 基于动态代理方式实现;aspectj 基于静态代理方式实现。 spring aop 仅支持方法级别的 pointcut;提供了完全的 aop 支持,它还支持属性级别的 pointcut。
6.8. 如何理解 spring 中的代理?
将 advice 应用于目标对象后创建的对象称为代理。在客户端对象的情况下,目标对象和代理对象是相同的。
advice + target object = proxy
6.9. 什么是编织(weaving)?
为了创建一个 advice 对象而链接一个 aspect 和其它应用类型或对象,称为编织(weaving)。在 spring aop 中,编织在运行时执行。请参考下图:
7. mvc
7.1. spring mvc 框架有什么用?
spring web mvc 框架提供 模型-视图-控制器 架构和随时可用的组件,用于开发灵活且松散耦合的 web 应用程序。 mvc 模式有助于分离应用程序的不同方面,如输入逻辑,业务逻辑和 ui 逻辑,同时在所有这些元素之间提供松散耦合。
7.2. 描述一下 dispatcherservlet 的工作流程
dispatcherservlet 的工作流程可以用一幅图来说明:
- 向服务器发送 http 请求,请求被前端控制器
dispatcherservlet
捕获。 dispatcherservlet
根据 -servlet.xml 中的配置对请求的 url 进行解析,得到请求资源标识符(uri)。然后根据该 uri,调用handlermapping
获得该 handler 配置的所有相关的对象(包括 handler 对象以及 handler 对象对应的拦截器),最后以handlerexecutionchain
对象的形式返回。dispatcherservlet
根据获得的handler
,选择一个合适的handleradapter
。(附注:如果成功获得handleradapter
后,此时将开始执行拦截器的 prehandler(...)方法)。-
提取
request
中的模型数据,填充handler
入参,开始执行handler
(controller
)。 在填充handler
的入参过程中,根据你的配置,spring 将帮你做一些额外的工作:handler(controller)执行完成后,向dispatcherservlet
返回一个modelandview
对象;- httpmessageconveter: 将请求消息(如 json、xml 等数据)转换成一个对象,将对象转换为指定的响应信息。
- 数据转换:对请求消息进行数据转换。如
string
转换成integer
、double
等。 - 数据根式化:对请求消息进行数据格式化。 如将字符串转换成格式化数字或格式化日期等。
- 数据验证: 验证数据的有效性(长度、格式等),验证结果存储到
bindingresult
或error
中。
- 根据返回的
modelandview
,选择一个适合的viewresolver
(必须是已经注册到 spring 容器中的viewresolver
)返回给dispatcherservlet
。 viewresolver
结合model
和view
,来渲染视图。- 视图负责将渲染结果返回给客户端。
7.3. 介绍一下 webapplicationcontext
webapplicationcontext 是 applicationcontext 的扩展。它具有 web 应用程序所需的一些额外功能。它与普通的 applicationcontext 在解析主题和决定与哪个 servlet 关联的能力方面有所不同。
免费java资料需要自己领取,涵盖了java、redis、mongodb、mysql、zookeeper、spring cloud、dubbo高并发分布式等教程,一共30g。
传送门: https://mp.weixin.qq.com/s/jzddfh-7ynudmkjt0irl8q