Writing by Tor小黑 —-心中藏之,无日忘之
Spring 框架两大核心机制(IoC、AOP)
- IoC(控制反转)/ DI(依赖注入)
- AOP(面向切面编程)
Spring 是一个企业级开发框架,是软件设计层面的框架,优势在于可以将应用程序进行分层,开发者可以自主选择组件。
MVC:Struts2、Spring MVC
ORMapping:Hibernate、MyBatis、Spring Data
如何使用 IoC
- 创建 Maven 工程,pom.xml 添加依赖
1 |
|
- 创建实体类 Student
1 | package com.southwind.entity; |
- 传统的开发方式,手动 new Student
1 | Student student = new Student(); |
- 通过 IoC 创建对象,在配置文件中添加需要管理的对象,XML 格式的配置文件,文件名可以自定义。
1 |
|
- 从 IoC 中获取对象,通过 id 获取。
1 | //加载配置文件 |
配置文件
通过配置
bean
标签来完成对象的管理。id
:对象名。class
:对象的模版类(所有交给 IoC 容器来管理的类必须有无参构造函数,因为 Spring 底层是通过反射机制来创建对象,调用的是无参构造)
对象的成员变量通过
property
标签完成赋值。name
:成员变量名。value
:成员变量值(基本数据类型,String 可以直接赋值,如果是其他引用类型,不能通过 value 赋值)ref
:将 IoC 中的另外一个 bean 赋给当前的成员变量(DI)
1
2
3
4
5
6
7
8
9
10
11<bean id="student" class="com.southwind.entity.Student">
<property name="id" value="1"></property>
<property name="name" value="张三"></property>
<property name="age" value="22"></property>
<property name="address" ref="address"></property>
</bean>
<bean id="address" class="com.southwind.entity.Address">
<property name="id" value="1"></property>
<property name="name" value="科技路"></property>
</bean>
IoC 底层原理
- 读取配置文件,解析 XML。
- 通过反射机制实例化配置文件中所配置所有的 bean。
1 | package com.southwind.ioc; |
通过运行时类获取 bean
1 | ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring.xml"); |
这种方式存在一个问题,配置文件中一个数据类型的对象只能有一个实例,否则会抛出异常,因为没有唯一的 bean。
通过有参构造创建 bean
- 在实体类中创建对应的有参构造函数。
- 配置文件
1 | <bean id="student3" class="com.southwind.entity.Student"> |
1 | <bean id="student3" class="com.southwind.entity.Student"> |
给 bean 注入集合
1 | <bean id="student" class="com.southwind.entity.Student"> |
scope 作用域
Spring 管理的 bean 是根据 scope 来生成的,表示 bean 的作用域,共4种,默认值是 singleton。
- singleton:单例,表示通过 IoC 容器获取的 bean 是唯一的。
- prototype:原型,表示通过 IoC 容器获取的 bean 是不同的。
- request:请求,表示在一次 HTTP 请求内有效。
- session:回话,表示在一个用户会话内有效。
request 和 session 只适用于 Web 项目,大多数情况下,使用单例和原型较多。
prototype 模式当业务代码获取 IoC 容器中的 bean 时,Spring 才去调用无参构造创建对应的 bean。
singleton 模式无论业务代码是否获取 IoC 容器中的 bean,Spring 在加载 spring.xml 时就会创建 bean。
Spring 的继承
与 Java 的继承不同,Java 是类层面的继承,子类可以继承父类的内部结构信息;Spring 是对象层面的继承,子对象可以继承父对象的属性值。
1 | <bean id="student2" class="com.southwind.entity.Student"> |
Spring 的继承关注点在于具体的对象,而不在于类,即不同的两个类的实例化对象可以完成继承,前提是子对象必须包含父对象的所有属性,同时可以在此基础上添加其他的属性。
Spring 的依赖
与继承类似,依赖也是描述 bean 和 bean 之间的一种关系,配置依赖之后,被依赖的 bean 一定先创建,再创建依赖的 bean,A 依赖于 B,先创建 B,再创建 A。
1 |
|
Spring 的 p 命名空间
p 命名空间是对 IoC / DI 的简化操作,使用 p 命名空间可以更加方便的完成 bean 的配置以及 bean 之间的依赖注入。
1 |
|
Spring 的工厂方法
IoC 通过工厂模式创建 bean 的方式有两种:
- 静态工厂方法
- 实例工厂方法
静态工厂方法
1 | package com.southwind.entity; |
1 | package com.southwind.factory; |
1 | <!-- 配置静态工厂创建 Car --> |
实例工厂方法
1 | package com.southwind.factory; |
1 | <!-- 配置实例工厂 bean --> |
IoC 自动装载(Autowire)
IoC 负责创建对象,DI 负责完成对象的依赖注入,通过配置 property 标签的 ref 属性来完成,同时 Spring 提供了另外一种更加简便的依赖注入方式:自动装载,不需要手动配置 property,IoC 容器会自动选择 bean 完成注入。
自动装载有两种方式:
- byName:通过属性名自动装载
- byType:通过属性的数据类型自动装载
byName
1 | <bean id="cars" class="com.southwind.entity.Car"> |
byType
1 | <bean id="car" class="com.southwind.entity.Car"> |
byType 需要注意,如果同时存在两个及以上的符合条件的 bean 时,自动装载会抛出异常。
AOP
AOP:Aspect Oriented Programming 面向切面编程。
AOP 的优点:
- 降低模块之间的耦合度。
- 使系统更容易扩展。
- 更好的代码复用。
- 非业务代码更加集中,不分散,便于统一管理。
- 业务代码更加简洁存粹,不参杂其他代码的影响。
AOP 是对面向对象编程的一个补充,在运行时,动态地将代码切入到类的指定方法、指定位置上的编程思想就是面向切面编程。将不同方法的同一个位置抽象成一个切面对象,对该切面对象进行编程就是 AOP。
如何使用?
- 创建 Maven 工程,pom.xml 添加
1 | <dependencies> |
- 创建一个计算器接口 Cal,定义4个方法。
1 | package com.southwind.utils; |
- 创建接口的实现类 CalImpl。
1 | package com.southwind.utils.impl; |
上述代码中,日志信息和业务逻辑的耦合性很高,不利于系统的维护,使用 AOP 可以进行优化,如何来实现 AOP?使用动态代理的方式来实现。
给业务代码找一个代理,打印日志信息的工作交个代理来做,这样的话业务代码就只需要关注自身的业务即可。
1 | package com.southwind.utils; |
以上是通过动态代理实现 AOP 的过程,比较复杂,不好理解,Spring 框架对 AOP 进行了封装,使用 Spring 框架可以用面向对象的思想来实现 AOP。
Spring 框架中不需要创建 InvocationHandler,只需要创建一个切面对象,将所有的非业务代码在切面对象中完成即可,Spring 框架底层会自动根据切面类以及目标类生成一个代理对象。
LoggerAspect
1 | package com.southwind.aop; |
LoggerAspect 类定义处添加的两个注解:
@Aspect
:表示该类是切面类。@Component
:将该类的对象注入到 IoC 容器。
具体方法处添加的注解:
@Before
:表示方法执行的具体位置和时机。
CalImpl 也需要添加 @Component
,交给 IoC 容器来管理。
1 | package com.southwind.utils.impl; |
spring.xml 中配置 AOP。
1 |
|
context:component-scan
将 com.southwind
包中的所有类进行扫描,如果该类同时添加了 @Component
,则将该类扫描到 IoC 容器中,即 IoC 管理它的对象。
aop:aspectj-autoproxy
让 Spring 框架结合切面类和目标类自动生成动态代理对象。
- 切面:横切关注点被模块化的抽象对象。
- 通知:切面对象完成的工作。
- 目标:被通知的对象,即被横切的对象。
- 代理:切面、通知、目标混合之后的对象。
- 连接点:通知要插入业务代码的具体位置。
- 切点:AOP 通过切点定位到连接点。