Spring中的AOP技术(一)

xiaoxiao2021-02-27  375

AOP技术是面向切面编程,采取横向抽取机制,取代了传统继承体系的重复性代码(性能监视、事务管理、安全检查、缓存)。Spring的AOP使用纯java语言编写,不需要专门的编译过程和类加载过程,在运行期采用动态代理方式向目标类织入增强的代码。

AOP的相关术语: JoinPoint(连接点):被拦截的点,在Spring中所谓的点就是方法,Spring值支持方法类型的连接点。 CutPoint(切入点):对哪些连接点进行拦截的定义。 Adivice(通知/增强):拦截到JoinPoint后要做的事就叫做通知。通知分为前置通知、后置通知、环绕通知、异常通知、最终通知。 Introduction(引介):引介是一种特殊的通知,在不修改代码的前提下,可以动态的为类添加属性和方法。 Target(目标):代理的目标对象。 Weaving(织入):把增强应用到目标对象来创建代理对象的过程。Spring采用动态代理织入,而AspectJ采用编译器织入和类装载期织入。 Proxy(代理):一个类被AOP增强后,就产生一个结果代理类 Aspect(切面):是切入点和通知(引介)的结合。

Spring中AOP技术的底层原理 JDK动态代理:对接口或实现接口的类进行动态代理 CGLib动态代理:对类进行代理。

Spring中的五种通知方式 Spring中的通知:(增强代码) 前置通知 org.springframework.aop.MethodBeforeAdvice * 在目标方法执行前实施增强 后置通知 org.springframework.aop.AfterReturningAdvice * 在目标方法执行后实施增强 环绕通知 org.aopalliance.intercept.MethodInterceptor * 在目标方法执行前后实施增强 异常抛出通知 org.springframework.aop.ThrowsAdvice * 在方法抛出异常后实施增强 引介通知 org.springframework.aop.IntroductionInterceptor * 在目标类中添加一些新的方法和属性

Spring的AOP开发 针对所有方法的增强:(不带有切点的切面): 1.导入jar包

2.编写代理的类

//代理的接口 package com.zhangyike.aop; public interface Person { public void eat(); public void study(); public void play(); } //接口的实现类 package com.zhangyike.aop; public class ImpPerson implements Person { @Override public void eat() { System.out.println("实现类中吃"); } @Override public void study() { System.out.println("实现类中学"); } @Override public void play() { System.out.println("实现类中玩"); } }

3.编写增强的代码:

package com.zhangyike.aop; import java.lang.reflect.Method; import org.springframework.aop.MethodBeforeAdvice; /* * 使用前置增强 */ public class AdvisorPerson implements MethodBeforeAdvice{ //被拦截的方法执行前,执行这个方法 /* * method:执行的方法 * args:参数 * target:目标对象 */ @Override public void before(Method method, Object[] arg1, Object arg2) throws Throwable { System.out.println(method.getName() + "拦截前被增强"); } }

4.编写xml配置文件:

<?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:context="http://www.springframework.org/schema/context" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <!-- 不带切点的切面分析 --> <!-- 定义被增强的类,接口的实现类 --> <bean id="impPerson" class="com.zhangyike.aop.ImpPerson"></bean> <!-- 定义通知增强 --> <bean id="advisorPerson" class="com.zhangyike.aop.AdvisorPerson"></bean> <!-- 用Spring配置方式生成代理,注意这里属性的名称是固定的,不能自定义,否则会出现加载配置文件失败的异常。 --> <bean id="personProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 设置目标对象 --> <property name="target" ref="impPerson"></property> <!-- 代理要实现的接口,value是接口的全路径 --> <property name="proxyInterfaces" value="com.zhangyike.aop.Person"></property> <!-- 设置增强,用value,而不是ref,不设置这个,不会增强 --> <property name="interceptorNames" value="advisorPerson"></property> <!-- 默认采用jdk动态代理方式,强制使用CGLib动态代理方式 --> <property name="optimize" value="true"></property> </bean> </beans>

5.编写测试类

package com.zhangyike.aop; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; @RunWith(SpringJUnit4ClassRunner.class) //xml文件要和测试类在同一包中 @ContextConfiguration("applicationContext.xml") public class DemoTest1 { @Autowired @Qualifier("personProxy") Person personProxy; @Test public void test1(){ personProxy.eat(); personProxy.study(); personProxy.play(); } }

六、测试结果: eat拦截前被增强 实现类中吃 study拦截前被增强 实现类中学 play拦截前被增强 实现类中玩

带有切点的切面:(针对目标对象的某些方法进行增强) PointcutAdvisor 接口: DefaultPointcutAdvisor 最常用的切面类型,它可以通过任意Pointcut和Advice 组合定义切面 RegexpMethodPointcutAdvisor 构造正则表达式切点切面

1.导入jar包

2.编写代理的类

//代理的接口 package com.zhangyike.aop; public interface Person { public void eat(); public void study(); public void play(); } //接口的实现类 package com.zhangyike.aop; public class ImpPerson implements Person { @Override public void eat() { System.out.println("实现类中吃"); } @Override public void study() { System.out.println("实现类中学"); } @Override public void play() { System.out.println("实现类中玩"); } }

3.编写增强类

package com.zhangyike.aop1; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; /* * 环绕通知 */ public class AdvisorPerson implements MethodInterceptor{ @Override public Object invoke(MethodInvocation arg0) throws Throwable { System.out.println("方法执行前"); Object result = arg0.proceed();//执行目标对象的方法 System.out.println("方法执行后"); return result; } }

4.在xml中配置增强代码

<?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:context="http://www.springframework.org/schema/context" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <!-- 带切点的切面分析 --> <!-- 定义被增强的类,也就是目标类,接口的实现类 --> <bean id="impPerson" class="com.zhangyike.aop1.ImpPerson"></bean> <!-- 定义通知增强的类 --> <bean id="advisorPerson" class="com.zhangyike.aop1.AdvisorPerson"></bean> <!-- 定义切点 --> <bean id="myjoinpoint" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"> <!-- 定义表达式,说明哪些方法被拦截 --> <!-- .:任意字符 *:任意个 <property name="pattern" value=".*"/> <property name="pattern" value="com\.zhangyike\.aop\.Person\.add.*"/> <property name="pattern" value=".*add.*"></property> --> <!-- 对一个方法进行拦截 <property name="pattern" value="*.eat.*"></property> --> <!-- 对吃、玩多个方法方法拦截 --> <property name="patterns" value=".*eat.*,.*play.*"></property> <!-- 设置增强的类 --> <property name="advice" ref="advisorPerson" ></property> </bean> <!-- 用Spring配置方式生成代理,并定义切面--> <bean id="personProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 设置目标对象 --> <property name="target" ref="impPerson"></property> <!-- 代理要实现的接口,value是接口的全路径 --> <property name="proxyInterfaces" value="com.zhangyike.aop1.Person"></property> <!-- 设置增强,用value,而不是ref,不设置这个,不会增强 --> <property name="interceptorNames" value="myjoinpoint"></property> <!-- 默认采用jdk动态代理方式,强制使用CGLib动态代理方式 --> <property name="optimize" value="true"></property> </bean> </beans>

5.测试类:

package com.zhangyike.aop1; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; @RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration("applicationContext.xml") public class DemoTest1 { @Autowired//自动注入 @Qualifier("personProxy")//按类型注入 Person person; @Test public void test1(){ person.eat(); System.out.println(); person.study(); System.out.println(); person.play(); System.out.println(); } }

六、执行结果 方法执行前 实现类中吃 方法执行后

实现类中学

方法执行前 实现类中玩 方法执行后

可见带有切面的切点对拦截的方法进行了增强,而未拦截的方法没有设置增强。

自动代理模式 前面两个例子中,每个代理都是通过ProxyFactoryBean织入切面代理,在实际开发中,非常多的Bean每个都配置ProxyFactoryBean开发维护量巨大。

自动创建代理:基于后处理bean,在bean的创建过程中完成增强,生成bean代理对象。 BeanNameAutoProxyCreator 根据Bean名称创建代理 DefaultAdvisorAutoProxyCreator 根据Advisor本身包含信息创建代理 * AnnotationAwareAspectJAutoProxyCreator 基于Bean中的

简单的demo: 一、导包 二、编写代理的目标类

//第一个要代理的目标类 package com.zhangyike.aop2; public class ImpPerson { public void eat() { System.out.println("人中吃........."); } public void study() { System.out.println("人中学........."); } public void play() { System.out.println("人中玩........."); } } //第二个要代理的目标类 package com.zhangyike.aop2; public class Animal { public void eat() { System.out.println("动物中吃........."); } public void study() { System.out.println("动物中学........."); } public void play() { System.out.println("动物中玩........."); } }

第三步、编写增强的代码

//前置增强 package com.zhangyike.aop2; import java.lang.reflect.Method; import org.springframework.aop.MethodBeforeAdvice; public class BeforeAdvicor implements MethodBeforeAdvice{ @Override public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable { System.out.println(arg0.getName() + "--增强前---" ); } } //后置增强 package com.zhangyike.aop2; import java.lang.reflect.Method; import org.springframework.aop.AfterReturningAdvice; public class AfterAdvicor implements AfterReturningAdvice{ @Override public void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable { System.out.println(arg1.getName() + "--后置增强..."); } } //环绕增强 package com.zhangyike.aop2; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; public class AroundAdvicor implements MethodInterceptor { @Override public Object invoke(MethodInvocation arg0) throws Throwable { System.out.println(arg0.getMethod().getName() + "..增强前..."); Object result = arg0.proceed();//执行方法 System.out.println(arg0.getMethod().getName() + "..增强后..."); return result; } }

第四步、生成代理,配置xml文件

<?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:context="http://www.springframework.org/schema/context" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <!-- 带切点的切面分析 --> <!-- 定义被增强的类,也就是目标类,接口的实现类 --> <bean id="impPerson" class="com.zhangyike.aop2.ImpPerson"></bean> <bean id="animal" class="com.zhangyike.aop2.Animal"></bean> <!-- 定义通知增强的类 --> <bean id="aroundAdvicor" class="com.zhangyike.aop2.AroundAdvicor"></bean> <bean id="afterAdvicor" class="com.zhangyike.aop2.AfterAdvicor"></bean> <bean id="beforeAdvicor" class="com.zhangyike.aop2.BeforeAdvicor"></bean> <!-- 注意:如果一个方法被三种方式分别增强,那么先执行前置增强、在执行环绕增强、最后执行后置增强 --> <!-- 定义第一个带有切点切面 --> <bean id="myjoinpoint1" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"> <!-- 对吃、玩多个方法方法拦截 --> <property name="pattern" value=".*eat.*"></property> <!-- 设置增强的类 --> <property name="advice" ref="afterAdvicor" ></property> </bean> <!-- 定义第二个带有切点切面 --> <bean id="myjoinpoint2" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"> <!-- 对吃、玩多个方法方法拦截 --> <property name="pattern" value=".*eat.*"></property> <!-- 设置增强的类 --> <property name="advice" ref="beforeAdvicor" ></property> </bean> <!-- 定义第三个带有切点切面 --> <bean id="myjoinpoint3" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"> <!-- 对吃、玩多个方法方法拦截 --> <property name="patterns" value=".*eat.*,.*study.*"></property> <!-- 设置增强的类 --> <property name="advice" ref="aroundAdvicor" ></property> </bean> <!-- 自动生成代理,没有被定义切点切面的方法将不会被增强,执行他自己本身的方法,定义切点切面的方法将执行定义后的方法--> <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"></bean> </beans>

第五步、测试类

package com.zhangyike.aop2; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; @RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration("applicationContext.xml") public class DemoTest1 { @Autowired @Qualifier("animal")//注入的是自己的bean对象,说明在bean的创建过程中完成的增强,生成的bean对象就是增强之后的对象了。 Animal animal; @Autowired @Qualifier("impPerson") ImpPerson person; @Test public void test1(){ animal.eat(); System.out.println(); animal.play(); System.out.println(); animal.study(); System.out.println(); person.eat(); System.out.println(); person.play(); System.out.println(); person.study(); System.out.println(); } }

第六步:测试结果 eat–增强前— eat..增强前… 动物中吃……… eat..增强后… eat–后置增强…

动物中玩………

study..增强前… 动物中学……… study..增强后…

eat–增强前— eat..增强前… 人中吃……… eat..增强后… eat–后置增强…

人中玩………

study..增强前… 人中学……… study..增强后…

本篇文章给出了三个Demo,分别是不带切点的切面、带切点的切面、 自动代理模式,前两种是ProxyFactoryBean代理,第三种是 DefaultAdvisorAutoProxyCreator,区别是: ProxyFattoryBean:先有被代理对象,将被代理对象存入到代理类中生成代理。 DefaultAdvisorAutoProxyCreator:在bean生成的时候,就产生了代理对象,将生成的代理对象返回,生成的bean就是代理对象。

转载请注明原文地址: https://www.6miu.com/read-5456.html

最新回复(0)