Spring:IOC和AOP
IOC(控制反转)和AOP(面向方面编程)作为Spring框架的两个核心,可以很好地实现解耦合。
Spring:IOC和AOP
IOC(控制反转)和AOP(面向方面编程)作为Spring框架的两个核心,可以很好地实现解耦合。
简单来说,Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架。
一. IOC
(1) 引入
如以前都是在【业务层】实例化【持久层】对象,耦合性强!
当持久层代码变动时,业务层代码也要改变! 所以有了IOC控制反转,即由IOC容器将持久层的对象注入到业务层;
(2) 定义
控制反转即主动变成被动;
IOC控制反转,即 主动new对象
变成由于 外部(IOC容器)去提供对象
,被IOC容器所管理的对象也被称为 bean
;
(3) 作用
解耦合,如果dao层有变化,则不需要改变service层代码!
(4) 实现
Spring技术对IOC思想进行了实现:
Spring提供了Core Container容器,称为IOC容器
,用来充当IOC思想中的外部;
即在Spring中,IOC可以理解为由主动new对象,转换为由IOC容器提供对象;
IOC会管理大量的对象(用到了 反射机制
,根据路径名获取类文件,进而完成对象创建,创建对象调用的是无参构造
);
IOC会管理对象的创建和初始化的过程;
IOC容器负责对象的创建、初始化等一些列工作,被创建或管理的对象在IOC容器中统称为
Bean
;
(5) DI依赖注入
在容器中建立Bean与Bean之间的依赖关系的过程,称为依赖注入!
Spring通过DI(依赖注入)实现IOC(控制反转);
效果:在需要使用Service对象时,里面的dao对象就已经由IOC容器生成并注入了!
目标:充分解耦;
总结:
- IOC :用对象的时候不用程序去new,而是由IOC容器提供;
- DI:当IOC中的Bean有关系时,会绑定依赖;
二. AOP
(1) 引入
在软件开发过程中,有一些逻辑横向遍布在各个业务模块中,像权限、监控、日志、事务、异常重试等等,所以造成代码分散且冗余度高,和业务代码混夹在一起, 写起来不够优雅,改起来更是一种折磨!为了解决这些问题,AOP(Aspect Oriented Programming:面向切面编程)也就应运而生了,它是一种编程思想,就像OOP(面向对象编程)也是一种编程思想,所以AOP不是某种语言或某个框架特有的,它实现的是将横向逻辑与业务逻辑解耦,实现对业务代码无侵入,从而让我们更专注于业务逻辑本身,本质是在不改变原有业务逻辑的情况下增强横切逻辑。
(2) 概念
AOP可以说是对OOP(面向对象编程) 的补充和完善;
面向对象就是将事物的特性和行为抽象为一个对象,如people类有身高、体重、年龄等属性,也有吃饭、睡觉等行为。把这些特性和行为封装成一个类,然后可以统一调用。
AOP(面向切面编程) 就是把一个单独的业务功能抽取出来,然后动态把这个功能切入到需要的方法(或行为)中;
即将功能先定义好,需要的地方才切入,这样便于减少系统的重复代码,降低模块间的耦合度。
作用:
- 降低耦合度
- 不改变原有代码的基础下,增加新功能;
Spring AOP就是基于动态代理
实现的,动态代理又利用了反射机制
;
使用场景:
如果有相同的、大量使用的功能在很多地方增强,适合AOP------------即简化共性功能的开发;
(3) Spring中的AOP
Spring理念:无侵入式编程,即不改变原有代码的基础上,为其增加新的功能;(使用了动态代理
)
AOP将业务逻辑组件和切面类都加入到容器中,通过@Aspect通知注解给切面类的目标方法标注在哪运行。
连接点:所有的原始方法;
切入点Pointcut:需要被功能来增强的方法(特殊的连接点);
通知advice:即抽取出来的增强的功能;
切面:绑定通知和切入点的关系,一个通知对应一个切入点;
总结:
抽取程序中要增强的部分作为一个通知方法;
执行对应通知的方法就是切入点;
将通知和切入点绑定的就是切面,切面就是描述在哪个切入点上执行哪些通知;
(4) AOP通知类型(绑定的类型)
@Before
:前置
@After
:后置 (不管抛不抛异常都会运行)
@AfterReturning
:返回后通知
@AfterThrowing
:抛出异常后通知
@Around
:环绕,用的最多;
在原始方法前执行前置,原始方法后面执行后置;
需要有东西来表示对原始方法的调用;此处需要抛异常,因为不知原始操作是否会有异常;
(5) @Pointcut 切入点类型
@Pointcut切点表达式非常丰富,可以将 方法(method)、类(class)、接口(interface)、包(package) 等作为切入点,非常灵活,常用的有@annotation
、execution
、@within 等方式,
@annotation
@annotation方式是指:切入点 是指定作用于 方法 上的注解,即被Spring扫描到方法上带有该注解 就会执行切面通知。
@Pointcut(value = "@annotation(com.tiangang.aop.MethodLog)")
public void pointCut() {
}
(6) 入门案例
需求:不改变切入点代码的前提下,执行update()时也打印系统时间; ;
思路分析:
1.导入AOP和Aspect的坐标
2.准备连接点(原始方法)
3.制作共性功能------通知和通知类;
4.定义切入点Pointcut;
5.绑定通知方法和切入点(切面)
0. 准备:
BookDao:
BookDao实现类:
目前只有执行save()方法才会打印系统时间;
导入aop和aspect包;
aop包默认依赖于spring-context,所以只需要导aspectjweaver包;
1. 定义通知类和通知方法;
2. 在通知类中描述和定义切入点 @pointcut,绑定方法!
用切入点表达式定义切入点,使用 @Pointcut
;
①在Execution中 绑定 实际的切入点方法(切入点表达式):
②定义一个私有方法pt(空壳),代表切入点;
3. 用注释绑定切入点和通知方法
此处让通知方法在切入点的前面执行,用@Before
,括号中写定义的切入点的私有方法pt();
此处通知方法即打印当前时间;
此处的切入点即update()方法;
4. 通知类:@Component + @Aspect
要能让Spring控制,先用@Componet
将通知类定义为bean,
然后用@Aspect
让通知类中的东西生效,让Spring知道这是AOP;
在@Pointcut
定义的切入点处 绑定 实际的切入方法 update()
!
5. SpringConfig:@EnableAspectJAuotoProxy
告诉Spring:这里用了注解开发的AOP,即去启动了@Aspect,而Aspect告诉Spring去识别通知类;
效果:执行 update() 时会先调用通知方法 method()打印时间!
(7) AOP的运行流程
1.启动 Spring容器
2.读取 所有【切面配置】中的pointcut
切入点;
3.初始化 bean;
判定 bean对应的类(如BookDaoImpl)中的原始方法
(如update)是否匹配到【通知类中】配置的切入点
※
匹配成功,创建原始对象(即目标对象)的代理对象
;
4.获取bean执行方法;
当匹配成功,获取的bean是代理对象,则用代理对象去调用原始方法与通知方法;
(即用代理对象去调用对应的方法(update),然后走增强的通知方法的操作;)
整个AOP的实现过程是由代理模式来进行的;
总结:如果配置的切入点能和类(BookDaoImpl)的方法匹配上,那么就创建代理对象,用代理对象去调用原始方法和增强的通知方法; 否则就只创建原始对象;
(8) AOP案例:测量业务层接口万次执行效率
参考:https://blog.csdn.net/scm_2008/article/details/128593857
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
所有评论(0)