JAVA-03-2023年210道面试题归纳之Java框架(一)(连载中)

54.Spring是什么?

轻量级的开源J2EE框架。它是一个容器框架,用来装javaBean(java对象),中间层框架(万能胶)可以起一个连接作用,比如说把Struts和hibernate粘合再一起运用,可以让我们的企业开发更快、更简洁,Spring是一个轻量级的控制反转(IOC)和面向切面(AOP)的容器框架

  • 从大小与开销两方面而言Spring都是轻量化级的。
  • 通过控制反转(IOC)的技术达到松耦合的的目的
  • 提供了面向切面编程的丰富支持,允许通过分离应用的业务逻辑与系统级服务进行内聚性的开发
  • 包含并管理应用(Bean)的配置和生命周期,这个意义是一个容器。
  • 将简单的组件配置、组合成为复杂的应用,这个意义上是一个框架。

55.谈谈你对AOP的理解

系统是由许多不同的组件所组成的,每一个组件各负责一块特定功能。除实现自身核心功能以外,这些组件还经常承担着额外的职责。例如日志、事物管理和安全这样的核心服务经常融入到自身具有核心业务逻辑的组件中去。这些系统服务经常被称为横切关注点,因为它们会跨越系统的多个组件。
当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。

日志代码往往水平的散布在所有对象层次中,而与它散布到的对象的核心功能毫无关系。

在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。

AOP:将程序中的交叉业务逻辑(比如安全,日志,事务等),封装成一个切面,然后注入到目标对象(具体业务逻辑)中去。AOP可以对某个对象或者某些对象的功能进行增强,比如对象中的方法进行增强,可以在执行某个方法之前额外的做一些事事情,在某个方法执行之后额外的一些事情

56.谈谈你对IOC的理解

容器概念、控制反转、依赖注入
IOC容器:实际上就是一个map(key,value),里面存储的是各个对象(在xml里配置的bean节点、@repository、@service、@controller、@component),在项目启动的时候会读取配置文件里面的bean节点,根据全限定类名使用反射对象放到map里、扫描到打上上述注解的类还是通过反射创建对象放到map里。

这个时候map就有各种对象了,接下来我们在代码里需要用到里面的对象时,再通过DI注入(autowrired、resource等注解,xml里bean节点内的ref属性,项目启动的时候就会读区xml节点ref属性根据id注入,也会扫描这些注解,根据类型或id注入;id就是对象名)

控制反转:
没有引入IOC容器之前,对象A依赖于对象B,那么对象在初始化或者运行到某一个点到时候,自己必须主动去创建对象B或者使用已创建的对象B。无论是创建还是使用对象B,控制权都在自己手上。

注入IOC容器之后,对象A与对象B之间失去了直接联系,当对象A运行到需要对象B的时候,IOC容器会主动创建一个对象B注入到对象B需要的地方

通过前后的对比,不难看出来:对象A获得依赖对象B的过程,由主动行为变为了被动行为,控制权点到为止了过来,这就是“控制反转”这个名称由来。

全部对象的控制权全部缴给了“第三方”IOC容器,所以,IOC容器成了整个系统的关键核心,它起到了一种类似“粘合剂”的作用,把系统中的所有对象粘合在一起发挥作用,如果没有这个“粘合剂”,对象与对象之间会彼此失去联系,这就是有人把IOC容器比喻成“粘合剂”的由来。

依赖注入:

“获得依赖对象的过程被反转了”。控制被反转之后,获得依赖对象的过程由自身管理变成了由IOC容器主动注入。依赖注入是实现IOC的方法,就是IOC容器在运行期间,动态的将某种依赖关系注入到对象之中。

57.解释下Spring支持的几种bean的作用域

  • singleton:默认,每个容器只有一个bena的实例,单例的模式由BeanFactory自身来维护。该对象的生命周期是SpringIOC容器一致的(但是在第一次被注入时才会创建)。
  • prototype:为每一个bean请求提供一个实例。在每次注入时都会创建一个新的对象
  • request:bean被定义在每个HTTP请求中创建一个单例对象,也就是说在单个请求都会复用这一个单例对象。
  • session:与request范围相似,确保每个session中有一个实例,在session过期后,bean会随之失效。
  • application:bean被定义在ServletContext的生命周期中复用一个单例对象。
  • websocket:bean被定义在websocket的生命周期中复用一个单例对象。
  • global-session:全局作用域,global-session和Portlet应用相关。当你的应用部署在Portlet容器中工作时,它包含很多portlet。如果你想要声明让所有的protlet共用全局的存储变量的话,那么这个全局变量需要存储在global-seesion中。全局作用域与Sevelt中的session作用域效果相同。

58.Spring事物的实现方式和原理以及隔离级别?

在使用Spring框架时,可以有两种使用事务的方式,一种是编程式的,一种是申明式的。
@Transactional注解就是申明式的。

首先,事务这个概念是数据库层面的,Spring只是基于数据库中的事务进行了扩展,以及提供了一些能让程序员更加方便操作事物的方式。

比如我们可以通过在某个方法上增加@Transactional注解,就可以开启事物,这个方法中所有的sql都会在一个事物中执行,统一成功或失败。

在一个方法上加了@Transactional注解后,Spring会基于这个类生成一个代理对象,会将这个代理对象作为bean,当在使用这个代理对象的方法时,如果这个方法上存在@Transactional注解,那么代理逻辑会先把事物的自动提交设置为false,然后再去执行原本的业务逻辑方法,如果执行业务逻辑方法没有出现异常,那么代理逻辑中就会将事物进行提交,如果执行业务逻辑方法出现了一查过,那么则会将事物进行回滚。

当然,针对哪些异常回滚事物是可以配置的,可以利用@Transactional注解中的rollbakcFor属性进行配置,默认情况下会对RuntimeException和Error进行回滚。

spirng事物隔离级别就是数据库的隔离级别:外加一个默认级别

  • read uncommited(未提交读)
  • read committed(提交读、不可重复读)
  • repeatable read(可重复读)
  • serializable(可串行化)

数据库的配置隔离级别是Read Commited,而Spring配置的隔离级别是Repeatable Read,请问这时隔离级别是以哪个为准?
以Spring配置的为准,如果Spring设置的隔离级别数据库不支持,效果取决于数据库

59.Spring事务传播机制

多个事务方法相互调用时,事务如何在这些方法间传播,方法A是一个事务的方法,方法A执行过程中调用了方法B,那么方法B有无事务以及方法B对事务的要求不同,都会对方法A的事务具体执行造成影响,同时方法A的事务对方法B的事物执行也有影响,这种影响具体是由什么就由两个方法所定义的事务传播类型所决定。

  1. REQUIRED(Spring默认的事务传播类型):如果当前没有事务,则自己新建一个事物,如果当前存在事务,则加入到这个事务
  2. SUPPORTS:当前存在事务,则加入当前事务,如果当前没有事务,就以非事务的方式执行
  3. MANDATORY:当前存在事务,则加入当前事务,如果当前不存在事务,则抛出异常。
  4. REQUIRES_NEW:创建一个新事物,如果存在当前事务,则挂起该事务。
  5. NOT_SUPPORTED:以非事务方式执行,如果当前存在事务,则挂起当前事务。
  6. NEVER:不使用事务,如果当前事务存在,则抛出异常
  7. NESTED:如果当前事务存在,则在嵌套事务中执行,否则REQUIRED到操作一样(开启一个事务)

60.Spring事务什么时候会失效?

spring事务的原理是AOP,进行了切面增强,那么失效的根本原因就是这个AOP不起作用了!常见情况有如下几种:

  1. 发生自调用,类里面使用this调用本类的方法(this通常省略),此时这个this对象不是代理类,而是UserService对象本身!
    • 解决办法很简单,让那个this变成UserService的代理类即可!
  2. 方法不是public的:@Transactional只能用于public的方法上,否则事务不会失效,如果要用在非public方法上,可以开启AspectJ代理模式。
  3. 数据库不支持事务
  4. 没有被spring管理
  5. 异常被吃掉,事务不会回滚(或抛出的异常没有被定义,默认为RuntimeException)

61.什么是bean的自动装配,有哪些方式?

开启自动装配,只需要在xml配置文件中定义autowrite属性

<bean id = "cutomer" class = "com.xx.xx.Customer" autowrite = "" />

autowrite属性有五种装配方式:

  • no - 缺省情况下,自动配置是通过“ref”属性手动设定。

手动装配:以value或ref的方式明确制定属性值都是手动装配。
需要通过‘ref’属性来链接bean。

  • byName - 根据bean的属性名称进行自动装配。

Cutomer的属性名是person,Spring会讲bean id为person的bean通过setter方式进行自动装配

<bean id = "cutomer" class = "com.xx.xx.Cutomer" autowrite = "byName"/>
<bean id = "person" class = "com.xx.xx.Person"/>
  • byType - 根据bean 的类型进行自动装配。

Cutomer的属性person的类型为Person,Spring会将Person类型通过setter方法进行自动装配。

<bean id = "cutomer" class = "com.xx.xx.Cutomer" autowrite = "byType"/>
<bean id = "person" class = "com.xx.xx.Person"/>
  • constructor - 类似byType,不过是应用于构造器的参数,如果一个bean与构造参数的类型相同,则会进行自动装配,否则导致异常。

Cutomer构造函数的参数person类型为Person,Spring会讲Person类型通过构造方法进行自动装配。

<bean id = "cutomer" class = "com.xx.xx.Cutomer" autowrite = "construtor"/>
<bean id = "person" class = "com.xx.xx.Person"/>
  • autodetect - 如果有默认的构造器,则通过constructor方式进行自动装配,否则使用byType方法进行自动装配。

如果有默认的构造器,则痛过constructor方式进行自动装配,否则使用byType方法进行自动装配。

@Autowrited自动装配bean,可以在字段、setter方法、构造函数上使用。

62.Spring中的bean创建的生命周期有哪些步骤

Spring中的一个Bean的创建大概分为以下几个步骤

  1. 推断构造方法
  2. 实例化
  3. 填充属性,也就是依赖注入
  4. 处理Aware回调
  5. 初始化前,处理@PostConstruct注解
  6. 初始化,处理initalizingBean接口
  7. 初始化后,进行AOP
    ![[Pasted image 20230430101152.png]]

63.Spring中的Bean是线程安全的么

Spring本身并没有针对Bean做线程安全的处理,所以:

  1. 如果Bean是无状态的,那么Bean则是线程安全的
  2. 如果Bean是有状态的,那么Bean则不是线程安全的

另外,Bean是不是线程安全,根Bean的作用域没有关系,Bean的作用域只是代表Bean 的生命周期范围,对于任何生命周期的Bean都是一个对象,这个对象是不是线程安的,还是得看这个Bean对象本身。

64.ApplicationContext和BeanFactory有什么区别

BeanFactory是Spring中非常核心的组件,表示bean工厂,可以生成bean,维护bean,而ApplicationContext继承了BeanFactory,所以ApplicationContext拥有BeanFactory所有的特点,也是一个Bean工厂,但是ApplicationContext除开继承了BeanFactory之外,还继承了诸如EnvironmentCapable、MessageSource、ApplicationEventPublisher等接口,从而ApplicationContext还有获取系统环境变量、国际化、事件发布等功能,这是BeanFacotry所不具备的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值