AOP是OOP的延续,是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。
什么是OOP?
什么是AOP?
AOP(Aspect Oriented Programming)面向切面编程:
- 可以说AOP是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。但当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。在OOP设计中,大量代码的重复不利于各个模块的重用。
- AOP技术与OOP相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。简单的说就是把我们程序重复的代码抽取出来,在需要执行的时候,使用动态代理的技术,在不修改源码的基础上,对我们的已有方法进行增强。
- 使用"横切"技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处基本相似,比如权限认证、日志、事物。AOP的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。
在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待,Struts2的拦截器设计就是基于AOP的思想,是个比较经典的例子。因此,利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
AOP的相关概念:
- 横切关注点:对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点
- Aspect(切面):通常是一个类,里面可以定义切入点和通知
- JointPoint(连接点):程序执行过程中明确的点,一般是方法的调用。被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器
- Advice(通知):AOP在特定的切入点上执行的增强处理,有before(前置),after(后置),afterReturning(最终),afterThrowing(异常),around(环绕)
- Pointcut(切入点):就是带有通知的连接点,在程序中主要体现为书写切入点表达式
- weave(织入):将切面应用到目标对象并导致代理对象创建的过程
- introduction(引入):在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段
- AOP代理(AOP Proxy):AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类
- 目标对象(Target Object): 包含连接点的对象。也被称作被通知或被代理对象。POJO
AOP 的作用及优势
作用:在程序运行期间,不修改源码对已有方法进行增强。
优势:减少重复代码
- 提高开发效率
- 维护方便
AOP 的实现方式
使用动态代理技术(利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行)
静态织入的方式(引入特定的语法创建“方面”,从而使得编译器可以在编译期间织入有关“方面”的代码)
Spring中AOP的四种Advice(通知)类型
-
前置通知类型:在目标方法前实施增强(如在方法执行前加个日志)
-
后置通知类型:在目标方法后实施增强(如在方法执行后加个日志)
-
环绕通知类型:在目标方法执行前后实施增强(如在方法执行前后加个日志)
-
异常通知类型:在方法出现异常之后抛出
-
引介通知类型:在目标方法中添加一些属性和方法
application.xml配置文件:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
//添加AOP的命名空间
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsd">
<bean id="productInfoService" class="com.tb.service.ProductInfoServiceImpl">
</bean>
<bean id="allLogAdvice" class="com.tb.aop.AllLogAdvice"></bean>
<!-- AOP切面编程:
<aop:aspect>只有一个通知和一个切入点
<aop:pointcut>切入点从具体的目标对象获取方法
-->
<aop:config>
<aop:aspect id="logaop" ref="allLogAdvice">
<aop:pointcut expression="execution(* com.tb.service.ProductInfoService.*(..))" id="logpoint"/>
<!--前置通知切入 -->
<aop:before method="beforeAdvice" pointcut-ref="logpoint"/>
<!--后置通知切入 -->
<aop:after-returning method="afterAdvice" pointcut-ref="logpoint"/>
<!--异常通知知切入 -->
<aop:after-throwing method="exceptionAdvice" pointcut-ref="logpoint" throwing="e"/>
<!--环绕通知切入 -->
<aop:around method="ArodAdvice" pointcut-ref="logpoint"/>
</aop:aspect>
</aop:config>
</beans>
使用AOP的几种方式
- 经典的基于代理的AOP
- @AspectJ注解驱动的切面
- 纯POJO切面
- 注入式AspectJ切面