后端研发Marion 发表于 2021-3-13 11:21:59

一文看懂Spring核心概念和源码解读

# 一文看懂Spring核心概念和源码解读

## 本文目次


[*]一、Spring框架的模块组成
[*]二、Spring的核心原理
[*]三、Spring的两大核心接口:BeanFactory和ApplicationContext
[*]四、Spring容器中的Bean
[*]五、单例Bean的线程安全和并发问题(重点)
[*]六、Spring的事务处理
[*]七、Spring框架中用到的设计模式
[*]八、Spring框架的变乱类型
## 一、Spring框架的模块组成

https://p3.pstatp.com/large/pgc-image/dd0c392eb5c04ae7b44ff1877d8edc03
1. 核心组件

1. 核心容器(Core Container)
2. AOP(Aspect Oriented Programming)
3. 设备支持(Instrument)
4. 数据访问及集成(Data Access/Integratioin)
5. Web
6. 报文发送(Messaging)
7. Test等模块。
2. Spring的优点

1. 低侵入式
2. DI机制减少组件耦合性
3. AOP
4. 扩展性
## 二、Spring的核心原理

Spring框架最核心的原理:IOC 和 AOP。IOC让相互协作的组件保持疏松的耦合,而AOP编程答应把遍布于应用各层的功能分离出来形成可重用的功能组件。
### 1. 控制反转 IOC

1. 容器控制创建对象,低沉对象之间耦合性,利于功能复用
2. 创建对象不须要new,而是利用JAVA反射机制根据配置文件动态创建对象和管理对象
3. Spring中三种注入方式:

[*]1. 构造器注入
[*]2. setter方法注入
[*]3. 基于注解注入(@Autowire)
### 2. 面向切面 AOP

1. AOP的理解

抽取并封装多个对象的公共行为和逻辑成一个可重用的模块`切面(Aspect)`


https://p3.pstatp.com/large/pgc-image/06056e34cd5f4bab89864663fb955780
2. 静态署理(AspectJ)和动态署理(Spring AOP)


[*]1. `静态署理`:AspectJ是静态署理的增强,编译时将AspectJ植入Java字节码中,运用时就是增强后的署理对象
[*]2. `动态署理`:动态署理不会修改Java字节码,是每次运行时在内存中临时为方法生成一个AOP对象,这个AOP对象包含了目标对象的全部方法,而且在特定的切点做了增强处理,并回调原对象的方法
3. Spring AOP动态署理的两种方式(JDK和CGLIB)


[*]1. `JDK`:JDK动态署理只提供接口的署理,不支持类的署理。
https://p1.pstatp.com/large/pgc-image/fb3091bc611448fbb151eada79f02ada

[*]2. `CGLIB`:是一个代码生成的类库,可以在运行时动态的生成指定类的一个子类对象,并覆盖此中特定方法并添加增强代码,从而实现AOP。


https://p1.pstatp.com/large/pgc-image/7828c3f16c1040acb1e2439bfd903975

[*]3. GLIB是通过继承的方式做的动态署理,因此如果某个类被标志为final,那么它是无法利用CGLIB做动态署理的。


https://p3.pstatp.com/large/pgc-image/215d548ad9664bfebcbc21269d13cd61


4. AOP的核心概念



https://p1.pstatp.com/large/pgc-image/bc0bf0eb6fa640e7ad07363ed0f71ba3


1. `切面(Aspect)`:被抽取的公共模块,可能会横切多个对象。 在Spring AOP中,切面可以利用通用类(基于模式的风格) 大概在普通类中以 @AspectJ 注解来实现。


https://p3.pstatp.com/large/pgc-image/06ea210f4c014f5c92f8ec8194e5d6f1


2. `连接点(Join point)`:指方法,在Spring AOP中,一个连接点 总是 代表一个方法的实验。
3.`关照(Advice)`:在切面的某个特定的连接点(Join point)上实验的动作。关照有各种类型,此中包括“around”、“before”和“after”等关照。许多AOP框架,包括Spring,都是以拦截器做关照模型, 并维护一个以连接点为中心的拦截器链。


https://p3.pstatp.com/large/pgc-image/4d80e9f6fc0f4d4c8b1578126a277a7a


https://p1.pstatp.com/large/pgc-image/0fbc445ed7674410a61170dd98a3c8c8


4. `切入点(Pointcut)`:切入点是指 我们要对哪些Join point进行拦截的定义。通过切入点表达式,指定拦截的方法,比如指定拦截add*、search*。


https://p3.pstatp.com/large/pgc-image/c5477d65e4c447d6820d03f9ae2bc14b


5. `引入(Introduction)`:(也被称为内部类型声明(inter-type declaration))。声明额外的方法大概某个类型的字段。Spring答应引入新的接口(以及一个对应的实现)到任何被署理的对象。例如,你可以利用一个引入来使bean实现 IsModified 接口,以便简化缓存机制。
6. `目标对象(Target Object)`: 被一个大概多个切面(aspect)所关照(advise)的对象。也有人把它叫做 被关照(adviced) 对象。 既然Spring AOP是通过运行时署理实现的,这个对象永久是一个 被署理(proxied) 对象。
7. `织入(Weaving)`:指把增强应用到目标对象来创建新的署理对象的过程。Spring是在运行时完成织入。
8. 切入点(pointcut)和连接点(join point)匹配的概念是AOP的关键,这使得AOP差别于其它仅仅提供拦截功能的旧技能。 切入点使得定位关照(advice)可独立于OO层次。
https://p1.pstatp.com/large/pgc-image/a8642b6638d44106ba9a17640aa2b008
5. AOP的应用场景



https://p3.pstatp.com/large/pgc-image/2c16a7ed34d04a21ae48fc22f4b96549
6. AOP的实战场景



https://p1.pstatp.com/large/pgc-image/792b83ea502b48b3b94125558f710005


https://p1.pstatp.com/large/pgc-image/a982720916304149b286368b9178a3d5


https://p3.pstatp.com/large/pgc-image/f6dd1a2248c54d3b9f78cde1289f1990


https://p3.pstatp.com/large/pgc-image/83a89f05dffa4a29b2c223170385bb91


https://p3.pstatp.com/large/pgc-image/0e92dc6aea584057afa0b74f02b00055


https://p3.pstatp.com/large/pgc-image/740d9c9e701042bf8fc11eb17824d82c


https://p3.pstatp.com/large/pgc-image/8908f2a16a094259a5b0c11553f6e4a1


https://p9.pstatp.com/large/pgc-image/5102dde13695472eb1dd4f4eac7f6b2d
7. 利用SpringAop的留意事项



https://p1.pstatp.com/large/pgc-image/86fa9bad311a404c9f0531256f6945b0


参考[探秘Spring AOP](https://www.imooc.com/learn/869)课程截图



https://p1.pstatp.com/large/pgc-image/c33d8a4ba3f044b490eae40f99bdeb5c


https://p9.pstatp.com/large/pgc-image/50d0a1917f824c379d4e565eb254bf26
https://p1.pstatp.com/large/pgc-image/1e2241cfb2d341b680d419d937659b0b


https://p1.pstatp.com/large/pgc-image/06cad3c897164e85a3dfa60ca413e085


https://p1.pstatp.com/large/pgc-image/69dc740a2ae3495c9588ec7438a8c743


https://p3.pstatp.com/large/pgc-image/6c66342ab4de43819599f9173128857c


https://p3.pstatp.com/large/pgc-image/59d068695bcf4ff4ad94af65b342216f
### 三、Spring的两大核心接口:BeanFactory和ApplicationContext

BeanFactory和ApplicationContext是Spring的两大核心接口,都可以当做Spring的容器。此中ApplicationContext是BeanFactory的子接口。
1. ApplicationContext 是 BeanFactory 的子接口,功能更全

[*]1. BeanFactory是Spring里面最底层的接口,包含了各种Bean的定义,读取bean配置文档,管理bean的加载、实例化,控制bean的生命周期,维护bean之间的依赖关系。ApplicationContext是BeanFactory的派生接口,除了提供BeanFactory所具有的功能外,还提供了更完备的框架功能:
[*]2. 继承MessageSource,因此支持国际化。
[*]3. 同一的资源文件访问方式。
[*]4. 提供在监听器中注册bean的变乱。
[*]5. 同时加载多个配置文件。
[*]6. 载入多个(有继承关系)上下文 ,使得每一个上下文都专注于一个特定的层次,比如应用的web层。
2. 加载方式

[*]1. `BeanFactroy` 采用的是延迟加载形式来注入Bean,即只有在利用到某个Bean时(调用getBean()),才对该Bean进行加载实例化。如许,我们就不能发现一些存在的Spring的配置问题。如果Bean的某一个属性没有注入,BeanFacotry加载后,直至第一次利用调用getBean方法才会抛出异常
[*]2. `ApplicationContext` 是在容器启动时,一次性创建了全部的Bean。如许,在容器启动时,我们就可以发现Spring中存在的配置错误,如许有利于查抄所依赖属性是否注入。 ApplicationContext启动后预载入全部的单实例Bean,通过预载入单实例bean ,确保当你须要的时候,你就不用等待,因为它们已经创建好了。
相对于基本的BeanFactory,ApplicationContext 唯一的不敷是占用内存空间。当应用程序配置Bean较多时,程序启动较慢。
3. 创建方式

[*]1. BeanFactory 通常以编程的方式被创建。
[*]2. ApplicationContext 除了编程方式,还能以声明的方式创建,如利用ContextLoader。
4. 注册方式
BeanFactory和ApplicationContext都支持BeanPostProcessor、BeanFactoryPostProcessor的利用,但两者之间的区别是:BeanFactory须要手动注册,而ApplicationContext则是自动注册。
### 四、Spring容器中的Bean

#### 1. Bean的作用域

1. `singleton(单例)`,在spring容器中仅存在一个Bean实例,Bean以单例方式存在,bean作用域范围的默认值。
2. `prototype(原型)`,每次从容器中调用Bean时,都返回一个新的实例,即每次调用getBean()时,相当于实验newXxxBean()。
3. `request`,每次HTTP请求都会创建一个新的Bean,该作用域仅实用于web的WebApplicationContext环境。
4. `session`,同一个HTTP Session共享一个Bean,差别Session利用差别的Bean。该作用域仅实用于web的WebApplicationContext环境。
5. `globalSession`,同一个全局 Session 共享一个 bean,用于 Portlet, 该作用域仅用于 WebApplication 环境。
#### 2. Bean注入的几种方式

1. 基于注解 @Autowired 的自动装配(最常用)

[*]1. authwire3个属性值:constructor、byName、byType
[*]2. 匹配多个Bean怎样注入:DefaultListableBeanFactory 的 determineAutowireCandidate
2. 基于构造方法注入
3. 基于Setter方法注入
#### 3. Bean的自动装配

在spring中,对象无需本身查找或创建与其关联的其他对象,由容器负责把须要相互协作的对象引用赋予各个对象,利用autowire来配置自动装载模式
1. 自动装配种类

[*]1. no:默认的方式是不进行自动装配的,通过手工设置ref属性来进行装配bean。
[*]2. byName:通过bean的名称进行自动装配,如果一个bean的 property 与另一bean 的name 相同,就进行自动装配。
[*]3. byType:通过参数的数据类型进行自动装配。
[*]4. constructor:利用构造函数进行装配,而且构造函数的参数通过byType进行装配。
[*]5. autodetect:自动探测,如果有构造方法,通过 construct的方式自动装配,否则利用
[*]6. byType的方式自动装配。
2. @Autowired和@Resource之间的区别

[*]1. @Autowired默认是按照类型装配注入的,默认环境下它要求依赖对象必须存在(可以设置它required属性为false)。
[*]2. @Resource默认是按照名称来装配注入的,只有当找不到与名称匹配的bean才会按照类型来装配注入。
#### 4. Bean的生命周期和加载流程(重点)

Spring Bean 的生命周期分为四个阶段和多个扩展点。扩展点又可以分为影响多个Bean和影响单个Bean。实例化和属性赋值对应构造方法和setter方法的注入,初始化和烧毁是用户能自定义扩展的两个阶段。


https://p3.pstatp.com/large/pgc-image/227cb544558b4a2da150fb5db172e63c


https://p1.pstatp.com/large/pgc-image/e4306cd2506543fe9f55d0a61112bcd0
1. 四个阶段(实例化 -> 属性赋值 -> 初始化 -> 烧毁)

[*]1. 实例化 Instantiation
[*]2. 属性赋值 Populate
[*]3. 初始化 Initialization
https://p1.pstatp.com/large/pgc-image/0c04813640834bee91df9b2f64a33769

[*]
[*]4. 烧毁 Destruction
2. 多个扩展点

[*]1. 影响多个Bean
1. BeanPostProcessor
2. InstantiationAwareBeanPostProcessor

[*]2. 影响单个Bean
1. Aware
1. Aware Group1
1. BeanNameAware
2. BeanClassLoaderAware
3. BeanFactoryAware
1. Aware Group2
1. EnvironmentAware
2. EmbeddedValueResolverAware
3. ApplicationContextAware(ResourceLoaderAware\ApplicationEventPublisherAware\MessageSourceAware)
3. 生命周期

[*]1. InitializingBean,初始化Bean
[*]2. DisposableBean,烧毁Bean
##### 4. 源码解读参考[一招带你搞定Spring源码,小白也能听懂的spring源码过程](https://www.ixigua.com/6898673954483864077)

1. Spring启动加载流程

[*]1. XML加载
[*]2. 解析
[*]3. 创建实例
[*]4. 利用
[*]5. 烧毁
2. 容器的作用

[*]1. 存放bean
[*]2. bean的数据布局
[*]3. list
[*]4. set
[*]5. map(K-V)
3. IOC容器

[*]1. `BeanDefinition`的定义信息
[*]2. 抽象定义规范接口`BeanDefinitionReader`
4. 创建对象方式

[*]1. new
[*]2. 反射怎样创建对象
[*]3. 工厂模式
[*]4. Spring中bean的默认scope是单例的
5. xml和注解中定义的是bean的定义信息,通过抽象定义接口解析到BeanDefinition
6. BeanDefinition通过new反射到实例化

[*]1. 实例化:在堆中开辟一块空间,属性都是默认值
[*]2. 初始化:给属性赋值
1. 填充属性
2. 实验初始化方法`init-method`
7. 初始化到完备对象
8. Spring的扩展性设计
9. BeanDefinition到BeanFactory
10. PostProcessor增强器,进行扩展实现
11. BeanFactoryPostProcessor

[*]1. 解析配置文件,处理占位符`placeholderConfigrerSupport`
[*]2. 可以自行继承PostProcessor
[*]3. `registerListeners`
12. bean的生命周期

[*]1. 实例化`反射`
[*]2. 填充属性`populateBean`
[*]3. 实验aware接口须要实现的方法`invokeAwareMethods`
[*]1. aware接口存在的意义是方便spring中的bean对象获取容器对象中的属性值
[*]4. BeanPostProcessor
[*]1. before
[*]2. init-method`invokeInitMethods`
[*]3. after
[*]4. aop扩展
[*]5. 完备对象
[*]6. 烧毁流程
13. Environment

[*]1. System.getEnv()
[*]2. System.getProperties()
14. 如果我想在spring生命周期的差别阶段做差别的处理工作?
1. 观察者模式:监听器、监听时间、多播器
15. spring中的do方法
## 五、单例Bean的线程安全和并发问题(重点)

单例Bean是Spring容器默认的方式,全部线程都共享一个单例实例Bean,确实会存在并发的问题。对于原型Bean,每次创建一个新对象,也就是线程之间并不存在Bean共享,天然是不会有线程安全的问题。
### 1、单例Bean的线程安全问题

1. 无状态对象(Stateless Bean)(`线程安全`)

[*]1. 没有实例变量的对象,不能生存数据,是不变类,是线程安全的。
[*]2. bean一旦实例化就被加进会话池中,各个用户都可以共用。即利用户已经灭亡,bean 的生命期也不一定结束,它可能依然存在于会话池中,供其他用户调用。
[*]3. 由于没有特定的用户,那么也就不能保持某一用户的状态,以是叫无状态bean。
[*]4. 无状态Bean,也就是线程中的操作不会对Bean的成员实验查询以外的操作,那么这个单例Bean是线程安全的。比如Spring mvc 的 Controller、Service、Dao等,这些Bean大多是无状态的,只关注于方法本身。
2. 有状态对象(Stateful Bean)(`线程不安全`) :

[*]1. 就是有实例变量的对象,可以生存数据,是非线程安全的。
[*]2. 每个用户有本身特有的一个实例,在用户的生存期内,bean保持了用户的信息,即“有状态”;一旦用户灭亡(调用结束或实例结束),bean的生命期也告结束。
[*]3. 即每个用户最初都会得到一个初始的bean。
### 2、线程安全问题的管理办法

1. 管理方案

[*]1. 只管避免在Bean对象中定义可变的成员变量,避免有状态的Bean。
[*]2. 如果确实须要利用状态的Bean,则在Bean对象中定义一个ThreadLocal成员变量,将须要的可变成员变量生存在ThreadLocal中。
[*]3. 也可以通过加锁的方法来管理线程安全,这种以时间换空间的场景在高并发场景下显然是不实际的,还是优先推荐利用ThreadLocal。
2. 多线程相同变量访问冲突问题:ThreadLocal和线程同步机制

[*]1. 线程同步:`时间换空间`
[*]2. ThreadLocal:`空间换时间`
[*]3. ThreadLocal会为每一个线程提供一个独立的变量副本,从而隔离了多个线程对数据的访问冲突。因为每一个线程都拥有本身的变量副本,从而也就没有须要对该变量进行同步了。
[*]4. ThreadLocal提供了线程安全的共享对象,在编写多线程代码时,可以把不安全的变量封装进ThreadLocal.
## 六、Spring的事务处理

### 1、Spring事务的种类

1. 编程式事务,利用TransactionTemplate。
2. 声明式事务,建立在AOP之上的。其本质是通过AOP功能,对方法前后进行拦截,将事务处理的功能编织到拦截的方法中,也就是在目标方法开始之前加入一个事务,在实验完目标方法之后根据实验环境提交大概回滚事务。
3. 区别:声明式事务最细粒度只能作用到方法级别,无法做到像编程式事务那样可以作用到代码块级别
### 2、Spring事务的隔离级别(5种)

1. 事务的隔离级别
### 3、Spring事务的传播行为(7种)

1. Spring事务的传播行为说的是,当多个事务同时存在的时候,Spring怎样处理这些事务的行为
## 七、Spring框架中用到的设计模式

### 1. 工厂模式

1. Spring利用工厂模式可以通过 BeanFactory 或 ApplicationContext 创建 bean 对象。两者的区别,

[*]1. BeanFactory :延迟注入(利用到某个 bean 的时候才会注入),占用更少的内存,程序启动速度更快。
[*]2. ApplicationContext :容器启动的时候,一次性创建全部 bean 。ApplicationContext除了BeanFactory的功能,另有额外更多功能,实际开发中,ApplicationContext更常用。
2. ApplicationContext的三个实现类:

[*]1. ClassPathXmlApplication:把上下文文件当成类路径资源。
[*]2. FileSystemXmlApplication:从文件系统中的 XML 文件载入上下文定义信息。
[*]3. XmlWebApplicationContext:从Web系统中的XML文件载入上下文定义信息。
### 2. 单例模式

Spring 中 Bean 的默认作用域就是 singleton(单例)的。利用单例模式的好处

[*]1. 对于频繁利用的对象,可以省略创建对象所花费的时间,这对于那些重量级对象而言,是非常可观的一笔系统开销;
[*]2. 由于 new 操作的次数减少,因而对系统内存的利用频率也会低沉,这将减轻 GC 压力,紧缩 GC 停顿时间。
### 3. 署理模式

Spring的AOP功能用到了JDK的动态署理和CGLIB字节码生成技能。
### 4. 模板方法

用来管理代码重复的问题。比如. RestTemplate, JmsTemplate, JpaTemplate等。
### 5. 观察者模式

定义对象键一种一对多的依赖关系,当一个对象的状态发生改变时,全部依赖于它的对象都会得到关照被制动更新,如Spring中listener的实现:ApplicationListener。
## 八、Spring框架的变乱类型

1. 上下文更新变乱(ContextRefreshedEvent)
2. 上下文开始变乱(ContextStartedEvent)
3. 上下文停止变乱(ContextStoppedEvent)
4. 上下文关闭变乱(ContextClosedEvent)
5. 请求处理变乱(RequestHandledEvent)
## 参考资料

- [一招带你搞定Spring源码,小白也能听懂的spring源码过程](https://www.ixigua.com/6898673954483864077)
- (https://www.imooc.com/learn/1108)
- [探秘Spring AOP](https://www.imooc.com/learn/869)
- (https://blog.csdn.net/shipfei_csdn/article/details/109530018)

驰光工作室 发表于 2021-3-14 07:54:29

很像是慕课的教程

好泰来 发表于 2021-3-13 19:30:27

详细...

小船掀大浪 发表于 2021-3-14 07:51:47

转发了

徐忠明勿忘初心 发表于 2021-3-14 11:37:00

转发了
页: [1]
查看完整版本: 一文看懂Spring核心概念和源码解读