spring aop切面实例理解
什么是AOP
AOP(Aspect-OrientedProgramming,面向方面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。
而AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。AOP代表的是一个横向的关系,如果说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为;那么面向方面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以获得其内部的消息。而剖开的切面,也就是所谓的“方面”了。然后它又以巧夺天功的妙手将这些剖开的切面复原,不留痕迹。
使用“横切”技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处都基本相似。比如权限认证、日志、事务处理。Aop 的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。正如Avanade公司的高级方案构架师Adam Magee所说,AOP的核心思想就是“将应用程序中的商业逻辑同对其提供支持的通用服务进行分离。”
实现AOP的技术,主要分为两大类:一是采用动态代理技术,利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行;二是采用静态织入的方式,引入特定的语法创建“方面”,从而使得编译器可以在编译期间织入有关“方面”的代码。
案例理解:
本次案例需要用到如下类
下面是配置文件
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" 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.xsd "> <!--.....................注入dao到service中以便进行下面操作......................--> <!--引入dao类--> <bean id="userDao" class="dao.impl.UserDaoImpl"></bean> <!--引入service类--> <bean id="userService" class="service.impl.UserServiceImpl"> <!--设置注入类--> <property name="dao" ref="userDao"/> </bean> <!--..................................A.O.P.开.始................................--> <!--引入增强类--> <bean id="theLogger" class="aop.UserServiceLogger"></bean> <!--配置aop增强信息--> <aop:config> <!--配置增强表达式,通过表达式expression寻找定义切入点--> <aop:pointcut id="pointcut" expression="execution(public void addNewUser(entity.User))"/> <!--配置切面,织入,ref织入的哪个类--> <aop:aspect ref="theLogger"> <!--前置增强,method增强方法,pointcut-ref增强表达式--> <aop:before method="before" pointcut-ref="pointcut"/> <!--后置增强,returning注入目标返回值--> <aop:after-returning method="afterReturning" pointcut-ref="pointcut" returning="result"/> </aop:aspect> </aop:config> </beans>
编写增强类
package aop; import org.aspectj.lang.JoinPoint; import java.util.Arrays; /** * 增强处理类 */ public class UserServiceLogger { /** * 前置增强 * @param jp 信息对象 */ public void before(JoinPoint jp){ System.out.println("调用"+jp.getTarget()+"的"+jp.getSignature()+ "方法,参数"+ Arrays.toString(jp.getArgs())); } /** * 后置增强 * @param jp 信息对象 * @param result 返回值 */ public void afterReturning(JoinPoint jp,Object result){ System.out.println("调用"+jp.getTarget()+"的"+jp.getSignature()+ "方法,返回值"+ result); } }
编写测试
@Test public void ts1(){ //获取配置文件对象实例 ApplicationContext ac=new ClassPathXmlApplicationContext("ApplicationContext.xml"); //获取userService对象实例 UserService userService=(UserService)ac.getBean("userService"); User use=new User(); use.setId(1); use.setUsername("test"); userService.addNewUser(use); }
允许如下:
不知道上面的案例和图你们理解了没有,下面是我根据自己的理解所画的图
根据本人的理解就是spring容器在我们调用方法时,在我们的方法之前调用了和在方法之后增强方法。
下面这篇详解AOP:https://blog.csdn.net/yuexianchang/article/details/77018603