Spring面试知识

1、说一下控制反转(IOC)

个人的理解:

IOC也叫控制反转,是一个工厂,工厂的目的是将对象的创建与使用分离开来,核心应用反射,实现这样解耦。在Spring中,就是将对象交由Spring容器管理,通过注解@Autowired和@Resource表示从Spring容器中找对应的对象,进行当前对象的注入,这样就实现依赖关系。

注解就是一种标记,表示具备什么样的属性,比如:是被Spring进行管理的、通过Spring注入的。

被注入的对象必须被下边的四个注解之一标注:

  • @Controller
  • @Service
  • @Repository
  • @Component

在Spring配置文件中配置 <context:annotation-config/>元素开启注解。还有一个概念是DI(依赖注入),和控制反转是同一个概念的不同角度的描述,即应用程序在运行时依赖IOC容器来动态注入对象需要的外部资源(对象等)。

知识点:

(1)Spring是什么?

Spring是一个轻量级的IOC和AOP容器框架。是为Java应用程序提供基础服务的一套框架,目的是用于简化企业应用程序的开发,它使得开发者只需要关心业务需求。

(2)Spring的核心模块有那些?

Spring的核心模块如下所示:

  • 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实现。

(3) Spring 有那些优点?

  • Spring的依赖注入将对象之间的依赖关系交给了框架来处理,减小了各个组件之间的耦合性;
  • AOP面向切面编程,可以将通用的任务抽取出来,复用性更高
  • Spring对于其余主流框架都提供了很好的支持,代码的侵入性很低

2、Spring中的AOP面向切面编程有了解吗?

AOP,是一个动态代理,将核心业务与代理业务分离出来,开发主要关心核心业务,代理业务(即切面)通过切入的方式,在真实业务前中后进行执行,这样实现核心业务的增强。常见的代理业务,有日志记录、权限判断等。

AOP中有如下的操作术语:通过切面中的切点,对切点的前中后,进行一个增强

Joinpoint(连接点): 类里面可以被增强的方法,这些方法称为连接点,切点选择部分或全部的连接点;

Aspect(切面):aspect关键字定义了一个类,这个类就是一个切面,是切点和增强结合。

Pointcut(切点):切面内部使用了pointcut定义切点,用于捕捉的需要应用切面的方法,这些方法也称为目标方法;所谓切入点是指我们要对哪些Joinpoint进行拦截的定义

Advice(通知/增强):通知就是那些需要在目标方法前中后执行的函数;所谓通知是指拦截到Joinpoint之后所要做的事情就是通知。

Weaving(织入):是把增强应用到目标方法的过程

Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下,Introduction可以在运行期为

 

定义切入点函数:

直接把execution已定义匹配表达式作为值传递给通知类型;

使用@Pointcut注解:

 

Spring中的AOP主要有两种实现方式:

  • 使用JDK动态代(dai)理实现,使用java.lang.reflection.Proxy类来处理
  • 使用cglib来实现

 

两种实现方式的不同之处:

JDK动态代理,只能对实现了接口的类生成代(dai)理,而不是针对类,该目标类型实现的接口都将被代(dai)理。原理是通过在运行期间创建一个接口的实现类来完成对目标对象的代(dai)理。实现步骤大概如下:

  • 定义一个实现接口InvocationHandler的类
  • 通过构造函数,注入被代(dai)理类
  • 实现invoke( Object proxy, Method method, Object[ ] args)方法
  • 在主函数中获得被代(dai)理类的类加载器
  • 使用Proxy.newProxyInstance( )产生一个代(dai)理对象
  • 通过代(dai)理对象调用各种方法

 

cglib主要针对类实现代(dai)理对是否实现接口无要求。原理是对指定的类生成一个子类,覆盖其中的方法,因为是继承,所以被代(dai)理的类或方法不可以声明为final类型。实现步骤大概如下:

  • 定义一个实现了MethodInterceptor接口的类
  • 实现其 intercept()方法,在其中调用proxy.invokeSuper( )

3、IOC容器的初始化过程?

我从设计的方面,IOC容器要进行对象的管理,肯定需要设计统一的数据结构进行存储管理的对象,初始化的过程就是将需要管理的对象转化为存储结构形式。就可以分为三个步骤:找到要管理的数据(资源定位),转化为内部的数据结构(BeanDefinition),最后进行存储(注册,将Bean Definition存在map里面,这样查找的时候就能在O(1)左右的时间复杂度内找到)。

  • Resource资源定位:

Resouce定位是指BeanDefinition的资源定位,也就是IOC容器找数据的过程。Spring中使用外部资源来描述一个Bean对象,IOC容器第一步就是需要定位Resource外部资源。由ResourceLoader通过统一的Resource接口来完成定位。
 

  • BeanDefinition的载入:

载入过程就是把定义好的Bean表示成IOC容器内部的数据结构,即BeanDefinition。在配置文件中每一个Bean都对应着一个BeanDefinition对象。

通过BeanDefinitionReader读取,解析Resource定位的资源,将用户定义好的Bean表示成IOC容器的内部数据结构BeanDefinition。

在IOC容器内部维护着一个BeanDefinitionMap的数据结构,通过BeanDefinitionMap,IOC容器可以对Bean进行更好的管理。
 

  • BeanDefinition的注册:

注册就是将前面的BeanDefition保存到Map中的过程,通过BeanDefinitionRegistry接口来实现注册。

(1)依赖注入发生的时刻?

主要看配置,lazy-init=false表示不开启延迟加载,在容器启动的时候即创建该Bean。对应的,我们还可以配置lazy-init=true表示开启延迟加载,那么该Bean的创建发生在应用程序第一次向容器索取Bean时,通过getBean()方法的调用完成。

(2)BeanFactory和FactoryBean的区别:

  • BeanFactory:Bean工厂,是一个工厂(Factory), 是Spring IOC容器的最顶层接口,它的作用是管理Bean,即实例化、定位、配置应用程序中的对象及建立这些对象间的依赖。
  • FactoryBean:工厂Bean,是一个Bean,作用是产生其他Bean实例,需要提供一个工厂方法,该方法用来返回其他Bean实例。

(3)BeanFactory和ApplicationContext有什么区别?

BeanFactory是Spring里面最顶层的接口,包含了各种Bean的定义,读取Bean配置文档,管理Bean的加载、实例化,控制Bean的生命周期,维护Bean之间的依赖关系。

ApplicationContext接口是BeanFactory的派生,除了提供BeanFactory所具有的功能外,还提供了更完整的框架功能:

  • 继承了MessageSource,支持国际化。
  • 提供了统一的资源文件访问方式。
  • 提供在Listener中注册Bean的事件。
  • 提供同时加载多个配置文件的功能。
  • 载入多个(有继承关系)上下文 ,使得每一个上下文都专注于一个特定的层次,比如应用的web层。

ApplicationContext 三种常见的实现方式:

  • FileSystemXmlApplicationContext:此容器从一个XML文件中加载Bean的定义,XML Bean 配置文件的全路径名必须提供给它的构造函数
  • ClassPathXmlApplicationContext:此容器也从一个XML文件中加载Bean的定义,需要正确设置classpath因为这个容器将在classpath里找Bean配置
  • WebXmlApplicationContext:此容器加载一个XML文件,定义了一个WEB应用的所有Bean。

在创建Bean和内存占用方面的区别:

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

BeanFactory和ApplicationContext的优缺点分析:

BeanFactory的优缺点:

优点:应用启动的时候占用资源很少,对资源要求较高的应用,比较有优势;

缺点:运行速度会相对来说慢一些。而且有可能会出现空指针异常的错误,而且通过Bean工厂创建的Bean生命周期会简单一些。
 

ApplicationContext的优缺点:

优点:所有的Bean在启动的时候都进行了加载,系统运行的速度快;在系统启动的时候,可以发现系统中的配置问题。

缺点:把费时的操作放到系统启动中完成,所有的对象都可以预加载,缺点就是内存占用较大。

 

4、Spring中Bean的作用域有哪几种?

Spring框架支持以下五种Bean的作用域:

  • singleton : Bean在每个Spring IOC 容器中只有一个实例,默认作用域。
  • prototype:一个Bean的定义可以有多个实例。
  • request:每次http请求都会创建一个Bean,该作用域仅在基于web的Spring ApplicationContext情形下有效。
  • session:在一个HTTP Session中,一个Bean定义对应一个实例。该作用域仅在基于web的Spring ApplicationContext情形下有效。
  • global-session:在一个全局的HTTP Session中,一个Bean对应一个实例。该作用域仅在基于web的Spring ApplicationContext情形下有效。

解析:

Spring中Bean的作用域常使用的是singleton,也就是单例作用域。那么单例Bean是线程安全的吗?

答: 单例Bean和线程安全与否没有必然的关系。多个线程在多个工作内存和主内存交互的时候会出现不一致的地方,那么就不是线程安全的。大部分的Spring Bean并没有可变的状态(比如Service类和DAO类),所以一定程度上可以说Spring的单例Bean是线程安全的。如果你的Bean有多种状态的话(比如 View Model 对象),就需要自行保证线程安全。在一般情况下,只有无状态的Bean才可以在多线程环境下共享。

(1)循环依赖了解吗?

我们再来看一个关于循环依赖的问题吧,面试官会这么问,“如果A对象创建的过程需要使用到B对象,但是B对象创建的时候也需要A对象,也就是构成了循环依赖的现象,那么Spring会如何解决?

答: 这是一种构造器循环依赖,通过构造器注入构成的循环依赖,此依赖是无法解决的,只能抛出BeanCurrentlyInCreationException异常表示循环依赖。

补充

5、Spring的事务有了解吗?

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

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

 

声明式事务的优点:

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

事务选择:

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

当多个Spring事务存在的时候,Spring定义了下边的7个传播行为来处理这些事务行为:

  • PROPAGATION_REQUIRED(propagation_required):如果当前存在事务,就加入该事务,如果当前没有事务,就创建一个新事务,该设置是最常用的设置。
  • PROPAGATION_SUPPORTS(supports):如果当前存在事务,就加入该事务,如果当前没有事务,就以非事务执行。
  • PROPAGATION_MANDATORY(mandatory):如果当前存在事务,就加入该事务,如果当前不存在事务,就抛出异常
  • PROPAGATION_REQUIRES_NEW(requires_new):创建新事务,无论当前存不存在事务,都创建新事务。
  • PROPAGATION_NOT_SUPPORTED(not_supported):以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
  • PROPAGATION_NEVER(never):以非事务方式执行,如果当前存在事务,则抛出异常。
  • PROPAGATION_NESTED(nested):如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则按REQUIRED属性执行。

解析:

Spring事务的考察,我们主要是掌握声明式和编程式事务管理的区别与选择。事务的传播行为也要有一个了解,另外,Spring事务的隔离级别和MySQL事务的隔离级别类似,依然存在读未提交,读已提交,可重复读以及串行四种隔离级别。 

 

6、SpringMVC的消息处理流程有了解吗?

SpringMVC是一种轻量级的Web层框架,是一个基于请求驱动的Web框架,使用了“前端控制器(DispatcherServlet)”模型来进行设计,再根据请求映射规则分发给相应的页面控制器进行处理。消息处理依次经过的组件如下:
DispatcherServlet、HandlerMapping、HandlerAdapter,返回一个ModelAndView逻辑视图名、ViewResolver、View

具体流程可以概括为:通过前端控制器DispatcherServlet来接收并且分发请求,然后通过HandlerMapping和HandlerAdapter找到具体可以处理该请求的Handler,经过逻辑处理,返回一个ModelAndView,经过ViewResolver处理,最后生成了一个View视图返回给了客户端。可以参考如下的示意图:

7、简单说下SpringBoot吧。

答: Spring Boot 是 Spring 开源组织下的子项目,是 Spring 组件一站式解决方案,主要是简化了使用 Spring 的难度,简省了繁重的配置,提供了各种启动器starter,开发者能快速上手。

SpringBoot的优点包括可以独立运行,简化了配置,可以实现自动配置,无代码生成以及XML配置,并且可以进行应用监控。

注解 @EnableAutoConfiguration, @Configuration, @ConditionalOnClass 就是自动配置的核心,实现了SpringBoot项目的自动配置。
 

SpringBoot的核心注解:

启动类上面的注解是@SpringBootApplication,它也是 Spring Boot 的核心注解,主要组合包含了以下 3 个注解:

  • @SpringBootConfiguration:组合了 @Configuration 注解,实现配置文件的功能。
  • @EnableAutoConfiguration:打开自动配置的功能,也可以关闭某个自动配置的选项。如关闭数据源自动配置功能: @SpringBootApplication(exclude = { DataSourceAutoConfiguration.class })。
  • @ComponentScan:Spring组件扫描。

 

SpringBoot项目启动分析:

Application类是通过SpringApplication类的静态run方法来启动应用的。打开这个静态方法,该静态方法真正执行的是两部分:new SpringApplication( )并且执行对象run()方法。
 

解析:

SpringBoot框架在平时使用较多,面试中也会考察其自动配置的实现原理以及项目的启动流程等知识点。

 

 

知识点:

轻量级和重量级概念的划分

  1.      主要看它使用了多少服务,使用的服务越多,容器为普通的java对象做的工作就越多,必然会影响到应用的发布时间,或者运行性能。
  2.      对于spring容器,他提供了很多服务,但默认是关闭的,应用需要某种服务,需要指明使用服务,如果服务很少,如只使用了spring核心服务,就认为此时应用属于轻量级的,如果使用了大部分的服务,这时就属于重量级的。
  3.      目前EJB容器就重量级。因为他默认为应用提供了Ejb规范中的所有功能,所以属于重量级。

 

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值