You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

151 lines
12 KiB
Markdown

2 years ago
# 第37讲 | 谈谈Spring Bean的生命周期和作用域
在企业应用软件开发中Java是毫无争议的主流语言开放的Java EE规范和强大的开源框架功不可没其中Spring毫无疑问已经成为企业软件开发的事实标准之一。今天这一讲我将补充Spring相关的典型面试问题并谈谈其部分设计细节。
今天我要问你的问题是谈谈Spring Bean的生命周期和作用域
## 典型回答
Spring Bean生命周期比较复杂可以分为创建和销毁两个过程。
首先创建Bean会经过一系列的步骤主要包括
* 实例化Bean对象。
* 设置Bean属性。
* 如果我们通过各种Aware接口声明了依赖关系则会注入Bean对容器基础设施层面的依赖。具体包括BeanNameAware、BeanFactoryAware和ApplicationContextAware分别会注入Bean ID、Bean Factory或者ApplicationContext。
* 调用BeanPostProcessor的前置初始化方法postProcessBeforeInitialization。
* 如果实现了InitializingBean接口则会调用afterPropertiesSet方法。
* 调用Bean自身定义的init方法。
* 调用BeanPostProcessor的后置初始化方法postProcessAfterInitialization。
* 创建过程完毕。
你可以参考下面示意图理解这个具体过程和先后顺序。
![](https://static001.geekbang.org/resource/image/3a/7e/3a51f06f56b905b8fbf1661359e1727e.png)
第二Spring Bean的销毁过程会依次调用DisposableBean的destroy方法和Bean自身定制的destroy方法。
Spring Bean有五个作用域其中最基础的有下面两种
* Singleton这是Spring的默认作用域也就是为每个IOC容器创建唯一的一个Bean实例。
* Prototype针对每个getBean请求容器都会单独创建一个Bean实例。
从Bean的特点来看Prototype适合有状态的Bean而Singleton则更适合无状态的情况。另外使用Prototype作用域需要经过仔细思考毕竟频繁创建和销毁Bean是有明显开销的。
如果是Web容器则支持另外三种作用域
* Request为每个HTTP请求创建单独的Bean实例。
* Session很显然Bean实例的作用域是Session范围。
* GlobalSession用于Portlet容器因为每个Portlet有单独的SessionGlobalSession提供一个全局性的HTTP Session。
## 考点分析
今天我选取的是一个入门性质的高频Spring面试题目我认为相比于记忆题目典型回答里的细节步骤理解和思考Bean生命周期所体现出来的Spring设计和机制更有意义。
你能看到Bean的生命周期是完全被容器所管理的从属性设置到各种依赖关系都是容器负责注入并进行各个阶段其他事宜的处理Spring容器为应用开发者定义了清晰的生命周期沟通界面。
如果从具体API设计和使用技巧来看还记得我在[专栏第13讲](http://time.geekbang.org/column/article/8471)提到过的Marker Interface吗Aware接口就是个典型应用例子Bean可以实现各种不同Aware的子接口为容器以Callback形式注入依赖对象提供了统一入口。
言归正传还是回到Spring的学习和面试。关于Spring也许一整本书都无法完整涵盖其内容专栏里我会有限地补充
* Spring的基础机制。
* Spring框架的涵盖范围。
* Spring AOP自身设计的一些细节前面[第24讲](http://time.geekbang.org/column/article/10076)偏重于底层实现原理这样还不够全面毕竟不管是动态代理还是字节码操纵都还只是基础更需要Spring层面对切面编程的支持。
## 知识扩展
首先我们先来看看Spring的基础机制至少你需要理解下面两个基本方面。
* 控制反转Inversion of Control或者也叫依赖注入Dependency Injection广泛应用于Spring框架之中可以有效地改善了模块之间的紧耦合问题。
从Bean创建过程可以看到它的依赖关系都是由容器负责注入具体实现方式包括带参数的构造函数、setter方法或者[AutoWired](https://docs.spring.io/spring-framework/docs/5.0.3.RELEASE/javadoc-api/org/springframework/beans/factory/annotation/Autowired.html)方式实现。
* AOP我们已经在前面接触过这种切面编程机制Spring框架中的事务、安全、日志等功能都依赖于AOP技术下面我会进一步介绍。
第二Spring到底是指什么
我前面谈到的Spring其实是狭义的[Spring Framework](https://github.com/spring-projects/spring-framework/blob/67ea4b3a050af3db5545f58ff85a0d132ee91c2a/spring-aop/src/main/java/org/aopalliance/aop/Advice.java)其内部包含了依赖注入、事件机制等核心模块也包括事务、O/R Mapping等功能组成的数据访问模块以及Spring MVC等Web框架和其他基础组件。
广义上的Spring已经成为了一个庞大的生态系统例如
* Spring Boot通过整合通用实践更加自动、智能的依赖管理等Spring Boot提供了各种典型应用领域的快速开发基础所以它是以应用为中心的一个框架集合。
* Spring Cloud可以看作是在Spring Boot基础上发展出的更加高层次的框架它提供了构建分布式系统的通用模式包含服务发现和服务注册、分布式配置管理、负载均衡、分布式诊断等各种子系统可以简化微服务系统的构建。
* 当然还有针对特定领域的Spring Security、Spring Data等。
上面的介绍比较笼统针对这么多内容如果将目标定得太过宽泛可能就迷失在Spring生态之中我建议还是深入你当前使用的模块如Spring MVC。并且从整体上把握主要前沿框架如Spring Cloud的应用范围和内部设计至少要了解主要组件和具体用途毕竟如何构建微服务等已经逐渐成为Java应用开发面试的热点之一。
第三我们来探讨一下更多有关Spring AOP自身设计和实现的细节。
先问一下自己,我们为什么需要切面编程呢?
切面编程落实到软件工程其实是为了更好地模块化而不仅仅是为了减少重复代码。通过AOP等机制我们可以把横跨多个不同模块的代码抽离出来让模块本身变得更加内聚进而业务开发者可以更加专注于业务逻辑本身。从迭代能力上来看我们可以通过切面的方式进行修改或者新增功能这种能力不管是在问题诊断还是产品能力扩展中都非常有用。
在之前的分析中我们已经分析了AOP Proxy的实现原理简单回顾一下它底层是基于JDK动态代理或者cglib字节码操纵等技术运行时动态生成被调用类型的子类等并实例化代理对象实际的方法调用会被代理给相应的代理对象。但是这并没有解释具体在AOP设计层面什么是切面如何定义切入点和切面行为呢
Spring AOP引入了其他几个关键概念
* Aspect通常叫作方面它是跨不同Java类层面的横切性逻辑。在实现形式上既可以是XML文件中配置的普通类也可以在类代码中用“@Aspect”注解去声明。在运行时Spring框架会创建类似[Advisor](https://github.com/spring-projects/spring-framework/blob/master/spring-aop/src/main/java/org/springframework/aop/Advisor.java)来指代它其内部会包括切入的时机Pointcut和切入的动作Advice
* Join Point它是Aspect可以切入的特定点在Spring里面只有方法可以作为Join Point。
* [](https://github.com/spring-projects/spring-framework/blob/67ea4b3a050af3db5545f58ff85a0d132ee91c2a/spring-aop/src/main/java/org/aopalliance/aop/Advice.java)[Advice](https://github.com/spring-projects/spring-framework/blob/67ea4b3a050af3db5545f58ff85a0d132ee91c2a/spring-aop/src/main/java/org/aopalliance/aop/Advice.java)它定义了切面中能够采取的动作。如果你去看Spring源码就会发现Advice、Join Point并没有定义在Spring自己的命名空间里这是因为他们是源自[AOP联盟](http://aopalliance.sourceforge.net/)可以看作是Java工程师在AOP层面沟通的通用规范。
Java核心类库中同样存在类似代码例如Java 9中引入的Flow API就是Reactive Stream规范的最小子集通过这种方式可以保证不同产品直接的无缝沟通促进了良好实践的推广。
具体的Spring Advice结构请参考下面的示意图。
![](https://static001.geekbang.org/resource/image/5b/ba/5b6955b4757c1a5fd0ecacdaf835e3ba.png)
其中BeforeAdvice和AfterAdvice包括它们的子接口是最简单的实现。而Interceptor则是所谓的拦截器用于拦截住方法也包括构造器调用事件进而采取相应动作所以Interceptor是覆盖住整个方法调用过程的Advice。通常将拦截器类型的Advice叫作Around在代码中可以使用“@Around”来标记或者在配置中使用“<aop:around>”。
如果从时序上来看,则可以参考下图,理解具体发生的时机。
![](https://static001.geekbang.org/resource/image/85/cb/85205c0c0ddcdafd2fad4ff5a53af0cb.png)
* Pointcut它负责具体定义Aspect被应用在哪些Join Point可以通过指定具体的类名和方法名来实现或者也可以使用正则表达式来定义条件。
你可以参看下面的示意图,来进一步理解上面这些抽象在逻辑上的意义。
![](https://static001.geekbang.org/resource/image/de/4a/dee96c33619d76d33281332bb3d2494a.png)
* Join Point仅仅是可利用的机会。
* Pointcut是解决了切面编程中的Where问题让程序可以知道哪些机会点可以应用某个切面动作。
* 而Advice则是明确了切面编程中的What也就是做什么同时通过指定Before、After或者Around定义了When也就是什么时候做。
在准备面试时如果在实践中使用过AOP是最好的否则你可以选择一个典型的AOP实例理解具体的实现语法细节因为在面试考察中也许会问到这些技术细节。
如果你有兴趣深入内部最好可以结合Bean生命周期理解Spring如何解析AOP相关的注解或者配置项何时何地使用到动态代理等机制。为了避免被庞杂的源码弄晕我建议你可以从比较精简的测试用例作为一个切入点如[CglibProxyTests](https://github.com/spring-projects/spring-framework/blob/da80502ea6ed4860f5bf7b668300644cdfe3bb5a/spring-context/src/test/java/org/springframework/aop/framework/CglibProxyTests.java)。
另外Spring框架本身功能点非常多AOP并不是它所支持的唯一切面技术它只能利用动态代理进行运行时编织而不能进行编译期的静态编织或者类加载期编织。例如在Java平台上我们可以使用Java Agent技术在类加载过程中对字节码进行操纵比如修改或者替换方法实现等。在Spring体系中如何做到类似功能呢你可以使用AspectJ它具有更加全面的能力当然使用也更加复杂。
今天我从一个常见的Spring面试题开始浅谈了Spring的基础机制探讨了Spring生态范围并且补充分析了部分AOP的设计细节希望对你有所帮助。
## 一课一练
关于今天我们讨论的题目你做到心中有数了吗今天的思考题是请介绍一下Spring声明式事务的实现机制可以考虑将具体过程画图。
请你在留言区写写你对这个问题的思考,我会选出经过认真思考的留言,送给你一份学习奖励礼券,欢迎你与我一起讨论。
你的朋友是不是也在准备面试呢?你可以“请朋友读”,把今天的题目分享给好友,或许你能帮到他。