1.Spring Aop简介
Spring AOP(Aspect-Oriented Programming,面向切面编程)是Spring框架中的一个重要特性。它为应用程序提供了一种以声明方式定义横切关注点(Cross-cutting Concerns)的能力,通过将这些关注点与主要业务逻辑进行解耦,提高了系统的可维护性和灵活性。
Spring AOP的工作原理
Spring AOP的工作原理可以简单描述为:
定义切面:开发者通过配置文件或者注解的方式定义切面(Aspect),切面包含横切关注点(Advice)和切入点(Pointcut)。
创建代理对象:Spring根据配置文件或者注解的规范,为被AOP增强的目标对象创建代理对象。
拦截方法调用:当方法调用到达切入点时,代理对象将会拦截方法调用,并执行相应的横切关注点。
执行增强逻辑:在拦截方法调用时,代理对象会根据配置执行相应的增强逻辑,如记录日志、事务管理等。
继续执行原始方法:在执行完增强逻辑后,代理对象会继续执行原始的方法逻辑,完成整个方法调用过程。
Spring AOP的类型
Spring AOP支持以下几种类型的横切关注点:
Before Advice:在目标方法调用之前执行的通知。
After Returning Advice:在目标方法成功返回后执行的通知。
After Throwing Advice:在目标方法抛出异常后执行的通知。
After Advice:无论目标方法如何结束,都会执行的通知。
Around Advice:在目标方法调用前后都可以自定义处理逻辑,并且决定是否继续执行目标方法。
Spring AOP的优势
降低代码的耦合度:通过将横切关注点与主要业务逻辑解耦,减少了代码的重复和侵入性。
提高了系统的可维护性:横切关注点的¥¥维护使得对整个系统的修改更加容易。
增强代码的灵活性:不同的切面可以通过配置的方式进行组合,实现不同的横切需求。
可以集成不同的框架:Spring AOP可以与其他框架如Spring MVC、Hibernate等进行集成,提供全方位的AOP功能。
总结起来,Spring AOP是Spring框架的一个重要特性,通过面向切面编程的方式,提供了一种以声明方式定义横切关注点的能力。它通过与主要业务逻辑解耦,降低了代码的耦合度,提高了系统的可维护性和灵活性。
2.AOP中关键性概念
在Spring AOP中,有几个关键性概念:
1.切面(Aspect):切面是一个类,它包含了一组通知和切点。通常,切面用于封装与横切关注点相关的行为。
连接点(Join Point):连接点是在应用程序执行过程中可以插入切面的点。在Spring AOP中,连接点可以是方法调用、方法执行、异常抛出等。
通知(Advice):通知是切面在连接点上执行的动作。Spring AOP支持几种类型的通知,包括前置通知(Before)、后置通知(After)、返回通知(After Returning)和异常通知(After Throwing)等。
切点(Pointcut):切点是一个表达式,它定义了在哪些连接点上应用通知。Spring AOP使用切点表达式来匹配连接点。
引入(Introduction):引入是一种在现有类中添加新方法或属性的方式,以增强该类的功能。Spring AOP允许通过引入来向现有类添加接口的实现。
织入(Weaving):织入是将切面应用到目标对象中的过程。Spring AOP支持编译时织入、类加载时织入和运行时织入。
这些概念是Spring AOP的核心组成部分,通过使用Spring AOP,可以实现横切关注点的模块化,并将其与应用程序的核心逻辑分离开来,提高代码的可维护性和可重用性。
3.Sping aop的专业术语
在Spring AOP中,还有一些专业术语,如下所示:
目标对象(Target Object):目标对象是被切面所通知的对象。它是应用程序中的实际业务对象。
代理对象(Proxy Object):代理对象是Spring AOP在运行时为目标对象创建的对象。代理对象包含了切面的通知逻辑,并在调用目标对象的方法之前或之后执行通知。
引入(Introduction):引入是一种在现有类中添加新方法或属性的方式,以增强该类的功能。Spring AOP允许通过引入来向现有类添加接口的实现。
织入(Weaving):织入是将切面应用到目标对象中的过程。Spring AOP支持编译时织入、类加载时织入和运行时织入。
切面优先级(Aspect Ordering):切面优先级定义了多个切面之间的执行顺序。在Spring AOP中,可以使用@Order注解或实现Ordered接口来指定切面的优先级。
切面自动代理(Aspect Auto-proxying):切面自动代理是Spring AOP的一种机制,它通过扫描应用程序上下文中的切面定义,并自动为匹配的目标对象创建代理对象。
切面表达式(Aspect Expression):切面表达式是用于定义切点的表达式。它可以使用类名、方法名、参数类型等来匹配连接点。
这些专业术语是Spring AOP中常用的概念,了解它们可以帮助更好地理解和使用Spring AOP框架。
4.spring的前置通知
伪代码论证
package com.zking.aop.advice; import java.lang.reflect.Method; import java.util.Arrays; import org.springframework.aop.MethodBeforeAdvice; /** * 买书、评论前加系统日志 * @author Administrator * */ public class MyMethodBeforeAdvice implements MethodBeforeAdvice { @Override public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable { // 在这里,可以获取到目标类的全路径及方法及方法参数,然后就可以将他们写到日志表里去 String target = arg2.getClass().getName(); String methodName = arg0.getName(); String args = Arrays.toString(arg1); System.out.println("【前置通知:系统日志】:"+target+"."+methodName+"("+args+")被调用了"); } }
spring_context.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" default-autowire="byType" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean class="com.zking.ioc.web.UserAction" id="userAction"> </bean> <bean class="com.zking.ioc.web.GoodsAction" id="goodsAction"> </bean> <bean class="com.zking.ioc.impl.UserServiceImpl1" id="userService"></bean> <bean class="com.zking.aop.biz.impl.BookBizImpl" id="bookBiz"></bean> <bean class="com.zking.aop.advice.MyMethodBeforeAdvice" id="methodBeforeAdvice"></bean> <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="methodPointcutAdvisor"> <property name="advice" ref="myAfterReturningAdvice"></property> <property name="pattern" value=".*buy"></property> </bean> <bean class="org.springframework.aop.framework.ProxyFactoryBean" id="bookProxy"> <property name="target" ref="bookBiz"></property> <property name="proxyInterfaces"> <list> <value>com.zking.aop.biz.IBookBiz</value> </list> </property> <property name="interceptorNames"> <list> <value>methodBeforeAdvice</value> <!-- <value>myAfterReturningAdvice</value>--> <value>methodPointcutAdvisor</value> <value>methodInterceptor</value> <value>myThrowsAdvice</value> </list> </property> </bean> </beans>
demo,后面几个都会用到demo就不重复了
package com.zking.demo; import com.zking.aop.biz.IBookBiz; import org.springframework.context.support.ClassPathXmlApplicationContext; public class demo1 { public static void main(String[] args) { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("sping-context.xml"); IBookBiz bookBiz = (IBookBiz) context.getBean("bookProxy"); bookBiz.buy("小朱","西医",9.9d); bookBiz.comment("你","你"); } }
5.Sping后置通知
package com.zking.aop.advice; import java.lang.reflect.Method; import java.util.Arrays; import org.springframework.aop.AfterReturningAdvice; /** * 买书返利 * @author Administrator * */ public class MyAfterReturningAdvice implements AfterReturningAdvice { @Override public void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable { String target = arg3.getClass().getName(); String methodName = arg1.getName(); String args = Arrays.toString(arg2); System.out.println("【后置通知:买书返利】:"+target+"."+methodName+"("+args+")被调用了,"+"该方法被调用后的返回值为:"+arg0); } }
spring_context.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" default-autowire="byType" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean class="com.zking.ioc.web.UserAction" id="userAction"> <!-- <property name="userService" ref="userService"></property>--> <!--<constructor-arg name="uname" value="袁辉sb"></constructor-arg>--> <!-- <constructor-arg name="age" value="11"></constructor-arg>--> <!-- <constructor-arg name="hobby" >--> <!-- <list>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- </list>--> <!-- </constructor-arg>--> </bean> <bean class="com.zking.ioc.web.GoodsAction" id="goodsAction"> <!-- <property name="userService" ref="userService"></property>--> <!-- <property name="gname" value="雨伞"></property>--> <!-- <property name="age" value="1"></property>--> <!-- <property name="peoples" >--> <!-- <list>--> <!-- <value>男的</value>--> <!-- <value>女的</value>--> <!-- </list>--> <!-- </property>--> </bean> <bean class="com.zking.ioc.impl.UserServiceImpl1" id="userService"></bean> <bean class="com.zking.aop.biz.impl.BookBizImpl" id="bookBiz"></bean> <bean class="com.zking.aop.advice.MyMethodBeforeAdvice" id="methodBeforeAdvice"></bean> <bean class="com.zking.aop.advice.MyAfterReturningAdvice" id="myAfterReturningAdvice"></bean> <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="methodPointcutAdvisor"> <property name="advice" ref="myAfterReturningAdvice"></property> <property name="pattern" value=".*buy"></property> </bean> <bean class="org.springframework.aop.framework.ProxyFactoryBean" id="bookProxy"> <property name="target" ref="bookBiz"></property> <property name="proxyInterfaces"> <list> <value>com.zking.aop.biz.IBookBiz</value> </list> </property> <property name="interceptorNames"> <list> <value>methodBeforeAdvice</value> <!-- <value>myAfterReturningAdvice</value>--> <value>methodPointcutAdvisor</value> <value>methodInterceptor</value> <value>myThrowsAdvice</value> </list> </property> </bean> </beans>
6.环绕通知
package com.zking.aop.advice; import java.util.Arrays; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; /** * 环绕通知 * 包含了前置和后置通知 * * @author Administrator * */ public class MyMethodInterceptor implements MethodInterceptor { @Override public Object invoke(MethodInvocation arg0) throws Throwable { String target = arg0.getThis().getClass().getName(); String methodName = arg0.getMethod().getName(); String args = Arrays.toString(arg0.getArguments()); System.out.println("【环绕通知调用前:】:"+target+"."+methodName+"("+args+")被调用了"); // arg0.proceed()就是目标对象的方法 Object proceed = arg0.proceed(); System.out.println("【环绕通知调用后:】:该方法被调用后的返回值为:"+proceed); return proceed; } }
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" default-autowire="byType" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean class="com.zking.ioc.web.UserAction" id="userAction"> <!-- <property name="userService" ref="userService"></property>--> <!--<constructor-arg name="uname" value="袁辉sb"></constructor-arg>--> <!-- <constructor-arg name="age" value="11"></constructor-arg>--> <!-- <constructor-arg name="hobby" >--> <!-- <list>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- </list>--> <!-- </constructor-arg>--> </bean> <bean class="com.zking.ioc.web.GoodsAction" id="goodsAction"> <!-- <property name="userService" ref="userService"></property>--> <!-- <property name="gname" value="雨伞"></property>--> <!-- <property name="age" value="1"></property>--> <!-- <property name="peoples" >--> <!-- <list>--> <!-- <value>男的</value>--> <!-- <value>女的</value>--> <!-- </list>--> <!-- </property>--> </bean> <bean class="com.zking.ioc.impl.UserServiceImpl1" id="userService"></bean> <bean class="com.zking.aop.biz.impl.BookBizImpl" id="bookBiz"></bean> <bean class="com.zking.aop.advice.MyMethodBeforeAdvice" id="methodBeforeAdvice"></bean> <bean class="com.zking.aop.advice.MyAfterReturningAdvice" id="myAfterReturningAdvice"></bean> <bean class="com.zking.aop.advice.MyMethodInterceptor" id="methodInterceptor"></bean> <bean class="com.zking.aop.advice.MyThrowsAdvice" id="myThrowsAdvice"></bean> <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="methodPointcutAdvisor"> <property name="advice" ref="myAfterReturningAdvice"></property> <property name="pattern" value=".*buy"></property> </bean> <bean class="org.springframework.aop.framework.ProxyFactoryBean" id="bookProxy"> <property name="target" ref="bookBiz"></property> <property name="proxyInterfaces"> <list> <value>com.zking.aop.biz.IBookBiz</value> </list> </property> <property name="interceptorNames"> <list> <value>methodBeforeAdvice</value> <!-- <value>myAfterReturningAdvice</value>--> <value>methodPointcutAdvisor</value> <value>methodInterceptor</value> <value>myThrowsAdvice</value> </list> </property> </bean> </beans>
7.sping异常通知
package com.zking.aop.advice; import org.springframework.aop.ThrowsAdvice; import com.zking.aop.exception.PriceException; /** * 出现异常执行系统提示,然后进行处理。价格异常为例 * @author Administrator * */ public class MyThrowsAdvice implements ThrowsAdvice { public void afterThrowing(PriceException ex) { System.out.println("【异常通知】:当价格发生异常,那么执行此处代码块!!!"); } }
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" default-autowire="byType" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean class="com.zking.ioc.web.UserAction" id="userAction"> <!-- <property name="userService" ref="userService"></property>--> <!--<constructor-arg name="uname" value="袁辉sb"></constructor-arg>--> <!-- <constructor-arg name="age" value="11"></constructor-arg>--> <!-- <constructor-arg name="hobby" >--> <!-- <list>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- </list>--> <!-- </constructor-arg>--> </bean> <bean class="com.zking.ioc.web.GoodsAction" id="goodsAction"> <!-- <property name="userService" ref="userService"></property>--> <!-- <property name="gname" value="雨伞"></property>--> <!-- <property name="age" value="1"></property>--> <!-- <property name="peoples" >--> <!-- <list>--> <!-- <value>男的</value>--> <!-- <value>女的</value>--> <!-- </list>--> <!-- </property>--> </bean> <bean class="com.zking.ioc.impl.UserServiceImpl1" id="userService"></bean> <bean class="com.zking.aop.biz.impl.BookBizImpl" id="bookBiz"></bean> <bean class="com.zking.aop.advice.MyMethodBeforeAdvice" id="methodBeforeAdvice"></bean> <bean class="com.zking.aop.advice.MyAfterReturningAdvice" id="myAfterReturningAdvice"></bean> <bean class="com.zking.aop.advice.MyMethodInterceptor" id="methodInterceptor"></bean> <bean class="com.zking.aop.advice.MyThrowsAdvice" id="myThrowsAdvice"></bean> <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="methodPointcutAdvisor"> <property name="advice" ref="myAfterReturningAdvice"></property> <property name="pattern" value=".*buy"></property> </bean> <bean class="org.springframework.aop.framework.ProxyFactoryBean" id="bookProxy"> <property name="target" ref="bookBiz"></property> <property name="proxyInterfaces"> <list> <value>com.zking.aop.biz.IBookBiz</value> </list> </property> <property name="interceptorNames"> <list> <value>methodBeforeAdvice</value> <!-- <value>myAfterReturningAdvice</value>--> <value>methodPointcutAdvisor</value> <value>methodInterceptor</value> <value>myThrowsAdvice</value> </list> </property> </bean> </beans>
8.Spring的过滤通知
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" default-autowire="byType" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean class="com.zking.ioc.web.UserAction" id="userAction"> <!-- <property name="userService" ref="userService"></property>--> <!--<constructor-arg name="uname" value="袁辉sb"></constructor-arg>--> <!-- <constructor-arg name="age" value="11"></constructor-arg>--> <!-- <constructor-arg name="hobby" >--> <!-- <list>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- <value>--> <!-- 农村--> <!-- </value>--> <!-- </list>--> <!-- </constructor-arg>--> </bean> <bean class="com.zking.ioc.web.GoodsAction" id="goodsAction"> <!-- <property name="userService" ref="userService"></property>--> <!-- <property name="gname" value="雨伞"></property>--> <!-- <property name="age" value="1"></property>--> <!-- <property name="peoples" >--> <!-- <list>--> <!-- <value>男的</value>--> <!-- <value>女的</value>--> <!-- </list>--> <!-- </property>--> </bean> <bean class="com.zking.ioc.impl.UserServiceImpl1" id="userService"></bean> <bean class="com.zking.aop.biz.impl.BookBizImpl" id="bookBiz"></bean> <bean class="com.zking.aop.advice.MyMethodBeforeAdvice" id="methodBeforeAdvice"></bean> <bean class="com.zking.aop.advice.MyAfterReturningAdvice" id="myAfterReturningAdvice"></bean> <bean class="com.zking.aop.advice.MyMethodInterceptor" id="methodInterceptor"></bean> <bean class="com.zking.aop.advice.MyThrowsAdvice" id="myThrowsAdvice"></bean> <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="methodPointcutAdvisor"> <property name="advice" ref="myAfterReturningAdvice"></property> <property name="pattern" value=".*buy"></property> </bean> <bean class="org.springframework.aop.framework.ProxyFactoryBean" id="bookProxy"> <property name="target" ref="bookBiz"></property> <property name="proxyInterfaces"> <list> <value>com.zking.aop.biz.IBookBiz</value> </list> </property> <property name="interceptorNames"> <list> <value>methodBeforeAdvice</value> <!-- <value>myAfterReturningAdvice</value>--> <value>methodPointcutAdvisor</value> <value>methodInterceptor</value> <value>myThrowsAdvice</value> </list> </property> </bean> </beans>
不管是前置通知、后置通知、环绕通知、异常通知、过滤通知,代码都是非业务核心代码,如日志、事务的管理(开启,提交、回滚)
总结:专业术语
目标对象 + 连接点 + 通知 = 代理
切入点是多个连接点的集合
通知 + 切入点= 适配器
总结:
通过本文的介绍,我们了解了Spring AOP的特点、专业术语以及不同类型的通知。Spring
AOP提供了一种优雅的方式来处理横切关注点,提高了代码的可维护性和可重用性。不同类型的通知可以满足不同的需求,开发人员可以根据具体场景选择合适的通知类型。通过深入学习和应用Spring
AOP,我们可以更好地设计和开发高质量的Java应用程序。