Spring AOP的几种实现方式总结
AOP核心概念
1、横切关注点
对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点
2、切面(aspect)
类是对物体特征的抽象,切面就是对横切关注点的抽象
3、连接点(joinpoint)
被拦截到的点,因为spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器
4、切入点(pointcut)
对连接点进行拦截的定义
5、通知(advice)
所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类
6、目标对象
代理的目标对象
7、织入(weave)
将切面应用到目标对象并导致代理对象创建的过程
8、引入(introduction)
在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段
Spring实现AOP所需要的包:
1、Spring提供的jar包
2、aopalliance.jar
3、aspectjweaver.jar
Spring实现AOP的方式:
1、Java动态代理
该方法针对接口的实例创建代理
applicationContext.xml的配置如下:
<?xmlversion="1.0"encoding="UTF-8"?> <beansxmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsd"> <beanid="concreteImplementor"class="com.marving.aop.ConcreteImplementor"/> <beanid="interceptorHandler"class="com.marving.aop.InterceptorHandler"/> <aop:config> <aop:aspectid="interceptor"ref="interceptorHandler"> <aop:pointcutid="addAllMethod"expression="execution(*com.marving.aop.Abstration.*(..))"/> <aop:beforemethod="doSomething"pointcut-ref="addAllMethod"/> <aop:aftermethod="doSomething"pointcut-ref="addAllMethod"/> </aop:aspect> </aop:config> </beans>
其中Abstration为接口,ConcreteImplementor为实现类,InterceptorHandler为代理拦截类。
publicinterface<spanstyle="font-size:12px;">Abstration</span>{ publicvoidoperation() }
//具体实现化角色 publicclassConcreteImplementorimplementsImplementor{ @Override publicvoidoperation(){ System.out.println("ConcreteImplementor"); } }
publicclassInterceptorHandler{ publicvoidprintTime(){ System.out.println("CurrentTime="+System.currentTimeMillis()); } }
2、CGLIB生成代理
CGLIB针对代理对象为类的情况使用。
通过实现MethodInterceptor接口,并实现publicObjectintercept(Objectobj,Methodm,Object[]args,MethodProxyproxy)throwsThrowable方法生成代理。
3、BeanNameAutoProxyCreator实现AOP
Spring为我们提供了自动代理机制,让容器为我们自动生成代理,把我们从烦琐的配置工作中解放出来,在内部,Spring使用BeanPostProcessor自动地完成这项工作。
具体配置如下:
<beanid="MyInterceptor"class="com.yesjpt.interceptor.MyInterceptor"></bean> <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> <propertyname="beanNames"> <list> <value>*Service</value> </list> </property> <propertyname="interceptorNames"> <list> <value>MyInterceptor</value> </list> </property> </bean>
其中*Service为需要拦截代理的bean,以Service结尾的都被拦截,并使用MyInterceptor进行拦截,可配置多个拦截器,按顺序执行。
importjava.lang.reflect.Method; importorg.aopalliance.intercept.MethodInterceptor; importorg.aopalliance.intercept.MethodInvocation; /** *@author * */ publicclassMyInterceptorimplementsMethodInterceptor{ @Override publicObjectinvoke(MethodInvocationinvocation)throwsThrowable{ Methodmethod=invocation.getMethod();//获取被拦截的方法 Object[]arguments=invocation.getArguments();//获取拦截方法的参数 /* *特殊,某些权限需要做特殊处理 *比如用户信息权限,在方法执行完毕返回的时候,要将电话号码与邮箱抹除 */ //环绕通知前置特殊处理 this.beforeReslove(); Objectproceed=invocation.proceed();//调用目标方法 //环绕通知后置特殊处理 proceed=this.afterReslove(); returnproceed; } privateObjectafterReslove(){ System.out.println("CurrentTime="+System.currentTimeMillis()); returnnull; } privatevoidbeforeReslove(){ System.out.println("CurrentTime="+System.currentTimeMillis()); } }
4、使用注解AspectJ实现AOP
ApplicationContext.xml加入
<aop:aspectj-autoproxy/>
创建切面处理类
packagecom.marving.aop; importjava.util.Arrays; importorg.aspectj.lang.ProceedingJoinPoint; importorg.aspectj.lang.annotation.Around; importorg.aspectj.lang.annotation.Aspect; importorg.aspectj.lang.annotation.Pointcut; importorg.springframework.stereotype.Component; @Aspect @Component publicclassAspectHandler{ @Pointcut("execution(*com.marving.service.BaseServ+.*(..))") privatevoiddoMethod(){ } /** *ThisisthemethodwhichIwouldliketoexecutebeforeaselectedmethod *execution. */ @Before("doMethod()") publicvoidbeforeAdvice(){ System.out.println("beforemethodinvoked."); } /** *ThisisthemethodwhichIwouldliketoexecuteafteraselectedmethod *execution. */ @After("doMethod()") publicvoidafterAdvice(){ System.out.println("aftermethodinvoked."); } //配置controller环绕通知,使用在方法aspect()上注册的切入点 @Around("doMethod()") publicObjectaround(ProceedingJoinPointpjp)throwsThrowable{ Objectresult=null; StringmethodName=pjp.getSignature().getName(); try{ System.out.println("Themethod["+methodName+"]beginswith"+Arrays.asList(pjp.getArgs())); result=pjp.proceed(); }catch(Throwablee){ System.out.println("Themethod["+methodName+"]occursexpection:"+e); thrownewRuntimeException(e); } System.out.println("Themethod["+methodName+"]ends"); returnresult; } }
通过表达式execution(*com.marving.service.BaseServ+.*(..))匹配切入点函数,并使用@Before@After@Around对所拦截方法执行前、中、后进行拦截并执行处理函数。
@Around@Before@After三个注解的区别@Before是在所拦截方法执行之前执行一段逻辑。@After是在所拦截方法执行之后执行一段逻辑。@Around是可以同时在所拦截方法的前后执行一段逻辑。
值得注意的是,Around在拦截方法后,需要返回一个方法执行结果,否则,原方法不能正常执行。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。