【Java面试专题】框架篇-01
Spring框架
单例bean是线程安全的吗?
不是线程安全的,Spring框架中有一个@Scope注解,默认的值就是singleton,单例的。因为一般在spring的bean的中都是注入无状态的对象,没有线程安全问题,如果在bean中定义了可修改的成员变量,是要考虑线程安全问题的,可以使用多例或者加锁来解决。
面试回答:
不是线程安全的。当多用户同时请求一个服务时,容器会给每个请求分配一个线程,这些线程会并发执行业务逻辑。如果处理逻辑中包含对单例状态的修改,比如修改单例的成员属性,就必须考虑线程同步问题。Spring框架本身并不对单例bean进行线程安全封装,线程安全和并发问题需要开发者自行处理。
通常在项目中使用的Spring bean是不可变状态(如Service类和DAO类),因此在某种程度上可以说Spring的单例bean是线程安全的。如果bean有多种状态(如ViewModel对象),就需要自行保证线程安全。最简单的解决办法是将单例bean的作用域由“singleton”变更为“prototype”。
关于prototype:
在 Spring 框架中,
prototype是 Bean 的作用域(Scope)之一。它表示每次从 Spring 容器中获取该 Bean 时,都会创建一个新的实例。这与默认的singleton作用域不同,singleton作用域下,Spring 容器中只会存在一个 Bean 实例。配置
prototype作用域
1. 使用注解
在类上使用
@Scope注解:2.2 使用 XML 配置
在 XML 配置文件中定义 Bean 时指定
scope属性:
什么是AOP?
AOP,即面向切面编程,在Spring中用于将那些与业务无关但对多个对象产生影响的公共行为和逻辑抽取出来,实现公共模块复用,降低耦合。常见的应用场景包括公共日志保存和事务处理。
使用场景:
记录操作日志、缓存、Spring实现的事务。
使用AOP来记录系统操作日志。主要思路是使用AOP的环绕通知和切点表达式,找到需要记录日志的方法,然后通过环绕通知的参数获取请求方法的参数,例如类信息、方法信息、注解、请求方式等,并将这些参数保存到数据库。
Spring中的事务是如何实现的?
Spring实现事务的本质是利用AOP完成的。它对方法前后进行拦截,在执行方法前开启事务,在执行完目标方法后根据执行情况提交或回滚事务。
Spring中事务失效的场景有哪些?
1. 如果方法内部捕获并处理了异常,没有将异常抛出,会导致事务失效。因此,处理异常后应该确保异常能够被抛出。
2. 如果方法抛出检查型异常(checked exception),并且没有在@Transactional注解上配置rollbackFor属性为Exception,那么异常发生时事务可能不会回滚。
3. 如果事务注解的方法不是公开(public)修饰的,也可能导致事务失效。
简单来说就是:异常捕获处理、抛出检查异常、非public方法
Spring的bean的生命周期
-
通过
BeanDefinition获取bean的定义信息。 -
调用构造函数实例化bean。
-
进行bean的依赖注入,例如通过setter方法或
@Autowired注解。 -
处理实现了
Aware接口的bean。 -
执行
BeanPostProcessor的前置处理器。 -
调用初始化方法,如实现了
InitializingBean接口或自定义的init-method。 -
执行
BeanPostProcessor的后置处理器,可能在这里产生代理对象。 -
最后是销毁bean。
- 点赞
- 收藏
- 关注作者
评论(0)