1 解析
1.1 使用global advisors demo
1.2 jdk代理和cglib代理的选择
1.3 如何强制使用CGLIB实现AOP?
1.4 JDK动态代理和CGLIB字节码生成的区别?
1.5 cglib的maven依赖
1.7 简化的proxy定义
1.8 父子bean demo
1.9 为什么要使用父子bean
2 代码演练
2.1 通配符使用 和 ProxyFactoryBean 连接点的选择
2.2 使用父子bean(parent标签使用)
1 解析
1.1 使用global advisors demo
用*做通配,匹配所有拦截器加入通知链,这种情况下只执行环绕通知
<!-- 执行所有sa打头的方法,都会引入切面 --> <!-- 实现之一:NameMatchMethodPointcut,根据方法名字进行匹配 ,自带方法 --> <bean id="pointcutBean" class="org.springframework.aop.support.NameMatchMethodPointcut"> <property name="mappedNames"> <list> <value>sa*</value> </list> </property> </bean> <bean id="defaultAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"> <property name="advice" ref="moocBeforeAdvice"></property> <property name="pointcut" ref="pointcutBean"></property> </bean> <!-- 从这里开始,引入多个通知 --> <!-- 引入的业务类不是ProxyFactoryBean ,而是getObject返回的,即 target==》bizLogicImplTarget --> <bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 具体业务类 --> <property name="target"> <ref bean="bizLogicImplTarget"/> </property> <!-- list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 --> <property name="interceptorNames"> <list> <!-- <value>defaultAdvisor</value> --> <!-- <value>moocAfterReturningAdvice</value> --> <!-- <value>moocMethodInterceptor</value> --> <!-- <value>moocThrowsAdvice</value> --> <value>mooc*</value> </list> </property> </bean>
1.2 jdk代理和cglib代理的选择
a 只有类,没有接口的话,使用cglib代理。(如2.2案例中没有bizLogic接口的话,会使用cglib代理)
b 有接口,使用jdk动态代理。
c 如果我们愿意,可以强制使用cglib代理。
1.3 如何强制使用CGLIB实现AOP?
1.3.1 在Spring3.2中
不用把CGLIB添加到classpath中,CGLIB被重新包装并包含在Spring核心的JAR(即基于CGLIB的AOP就像JDK动态代理一样“开箱即用”)
1.3.1 在Spring3.2之前的版本
* 添加CGLIB库,SPRING_HOME/cglib/*.jar
* 在spring配置文件中加入<aop:aspectj-autoproxy proxy-target-class="true"/>
1.3.2 在spring3.2之后的版本
没试过
1.4 JDK动态代理和CGLIB字节码生成的区别?
* JDK动态代理只能对实现了接口的类生成代理,而不能针对类
* CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法 因为是继承,所以该类或方法最好不要声明成final
1.5 cglib的maven依赖
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>2.2.2</version>
</dependency>
1.6 CGLIB代理的工作原理
CGLIB代理的工作原理是在运行时生成目标类的子类,Spring配置这个生成的子类委托方法调用到原来的目标,子类是用来实现Decorator模式,织入通知
1.7 简化的proxy定义
使用父子bean定义,以及内部bean定义,可能会带来更清洁和更简洁的代理定义(抽象属性标记父bean定义为抽象的这样它不能被实例化)
1.8 父子bean demo
<bean id="baseProxyBean" class="org.springframework.aop.framework.ProxyFactoryBean" lazy-init="true" abstract="true"> </bean> <bean id="bizLogicImpl" parent="baseProxyBean"> <!-- 具体业务类 --> <property name="target"> <ref bean="bizLogicImplTarget"/> </property> <property name="proxyInterfaces"> <value>com.imooc.aop.api.BizLogic</value> </property> <!-- list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 --> <property name="interceptorNames"> <list> <value>moocBeforeAdvice</value> <value>moocAfterReturningAdvice</value> <value>moocMethodInterceptor</value> <value>moocThrowsAdvice</value> <!-- <value>mooc*</value> --> </list> </property> </bean>
1.9 为什么要使用父子bean
用父子bean可以简化proxy定义,多个需要被代理的类的情况下使用父子 bean 方法会使代码更好复用
2 代码演练
2.1 通配符使用 和 ProxyFactoryBean 连接点的选择
测试类:
package com.imooc.test.aop; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.BlockJUnit4ClassRunner; import com.imooc.aop.api.BizLogic; import com.imooc.test.base.UnitTestBase; @RunWith(BlockJUnit4ClassRunner.class) public class TestAOPAPI extends UnitTestBase{ public TestAOPAPI() { super("classpath:spring-aop-api.xml"); // TODO Auto-generated constructor stub } @Test public void testSave(){ BizLogic bLogic = super.getbean("bizLogicImpl"); bLogic.sove(); } }
配置文件:
<?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" 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/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd 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" > <bean id="moocBeforeAdvice" class="com.imooc.aop.api.MoocBeforeAdvice"></bean> <bean id="moocAfterReturningAdvice" class="com.imooc.aop.api.MoocAfterReturningAdvice"></bean> <bean id="moocMethodInterceptor" class="com.imooc.aop.api.MoocMethodInterceptor"></bean> <bean id="moocThrowsAdvice" class="com.imooc.aop.api.MoocThrowsAdvice"></bean>
<bean id="bizLogicImplTarget" class="com.imooc.aop.api.BizLogicImpl"></bean> <!-- 从这里开始,引入多个通知 --> <!-- 引入的业务类不是ProxyFactoryBean ,而是getObject返回的,即 target==》bizLogicImplTarget --> <bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 具体业务类 --> <property name="target"> <ref bean="bizLogicImplTarget"/> </property> <!-- list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 --> <property name="interceptorNames"> <list> <!-- <value>defaultAdvisor</value> --> <!-- <value>moocAfterReturningAdvice</value> --> <!-- <value>moocMethodInterceptor</value> --> <!-- <value>moocThrowsAdvice</value> --> <value>mooc*</value> </list> </property> </bean> </beans>
实现类:
package com.imooc.aop.api; public class BizLogicImpl implements BizLogic{ @Override public String sove() { System.out.println("syso save"); return "BizLogicImpl save"; } }
接口:
package com.imooc.aop.api; public interface BizLogic { String sove(); }
环绕通知:
package com.imooc.aop.api; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; /** * 引入环绕通知 * * 实现 MethodInterceptor 接口 * @author weijingli * */ public class MoocMethodInterceptor implements MethodInterceptor{ @Override public Object invoke(MethodInvocation methodinvocation) throws Throwable { //实现前置通知的方法 //得到方法 得到类 System.out.println("MoocMethodInterceptor1:"+methodinvocation.getMethod().getName()+" " +methodinvocation.getStaticPart().getClass().getName()); Object obj = methodinvocation.proceed(); //实现后置通知方法 System.out.println("MoocMethodInterceptor2:"+obj); return obj; } }
打印日志:
四月 23, 2019 6:56:02 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh 信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Tue Apr 23 18:56:02 CST 2019]; root of context hierarchy 四月 23, 2019 6:56:02 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions 信息: Loading XML bean definitions from class path resource [spring-aop-api.xml] MoocMethodInterceptor1:sove java.lang.reflect.Method syso save MoocMethodInterceptor2:BizLogicImpl save 四月 23, 2019 6:56:04 下午 org.springframework.context.support.ClassPathXmlApplicationContext doClose 信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Tue Apr 23 18:56:02 CST 2019]; root of context hierarchy
2.2 使用父子bean(parent标签使用)
测试类:
package com.imooc.test.aop; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.BlockJUnit4ClassRunner; import com.imooc.aop.api.BizLogic; import com.imooc.test.base.UnitTestBase; @RunWith(BlockJUnit4ClassRunner.class) public class TestAOPAPI extends UnitTestBase{ public TestAOPAPI() { super("classpath:spring-aop-api.xml"); // TODO Auto-generated constructor stub } @Test public void testSave(){ BizLogic bLogic = super.getbean("bizLogicImpl"); bLogic.sove(); } }
配置文件:
<?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" 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/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd 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" > <bean id="moocBeforeAdvice" class="com.imooc.aop.api.MoocBeforeAdvice"></bean> <bean id="moocAfterReturningAdvice" class="com.imooc.aop.api.MoocAfterReturningAdvice"></bean> <bean id="moocMethodInterceptor" class="com.imooc.aop.api.MoocMethodInterceptor"></bean> <bean id="moocThrowsAdvice" class="com.imooc.aop.api.MoocThrowsAdvice"></bean> <bean id="bizLogicImplTarget" class="com.imooc.aop.api.BizLogicImpl"></bean> <!-- 执行所有sa打头的方法,都会引入切面 --> <!-- 实现之一:NameMatchMethodPointcut,根据方法名字进行匹配 ,自带方法 --> <bean id="pointcutBean" class="org.springframework.aop.support.NameMatchMethodPointcut"> <property name="mappedNames"> <list> <value>sa*</value> </list> </property> </bean> <bean id="defaultAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"> <property name="advice" ref="moocBeforeAdvice"></property> <property name="pointcut" ref="pointcutBean"></property> </bean> <!-- 从这里开始,引入多个通知 --> <!-- 引入的业务类不是ProxyFactoryBean ,而是getObject返回的,即 target==》bizLogicImplTarget --> <!-- <bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean"> --> <!-- 具体业务类 --> <!-- <property name="target"> --> <!-- <ref bean="bizLogicImplTarget"/> --> <!-- </property> --> <!-- list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 --> <!-- <property name="interceptorNames"> --> <!-- <list> --> <!-- <value>defaultAdvisor</value> --> <!-- <value>moocAfterReturningAdvice</value> --> <!-- <value>moocMethodInterceptor</value> --> <!-- <value>moocThrowsAdvice</value> --> <!-- <value>mooc*</value> --> <!-- </list> --> <!-- </property> --> <!-- </bean> --> <bean id="baseProxyBean" class="org.springframework.aop.framework.ProxyFactoryBean" lazy-init="true" abstract="true"> </bean>
<bean id="bizLogicImpl" parent="baseProxyBean"> <!-- 具体业务类 --> <property name="target"> <ref bean="bizLogicImplTarget"/> </property> <property name="proxyInterfaces"> <value>com.imooc.aop.api.BizLogic</value> </property> <!-- list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 --> <property name="interceptorNames"> <list> <value>moocBeforeAdvice</value> <value>moocAfterReturningAdvice</value> <value>moocMethodInterceptor</value> <value>moocThrowsAdvice</value> <!-- <value>mooc*</value> --> </list> </property> </bean> </beans>
实现类:
package com.imooc.aop.api; public class BizLogicImpl implements BizLogic{ @Override public String sove() { System.out.println("syso save"); return "BizLogicImpl save"; } }
接口:
package com.imooc.aop.api; public interface BizLogic { String sove(); }
前置通知:
package com.imooc.aop.api; import java.lang.reflect.Method; import org.springframework.aop.MethodBeforeAdvice; public class MoocBeforeAdvice implements MethodBeforeAdvice{ /** * 实现MethodBeforeAdvice接口,可以在连接点之前执行 * * * 作用是什么? * 将前一章中的xml配置方式改为了实现接口的实现方式 具体细节和使用参考开发文档 * */ @Override public void before(Method method, Object[] aobj, Object obj) throws Throwable { System.out.println("MoocBeforeAdvice:"+method.getName()+" "+ obj.getClass().getName()); } }
环绕通知:
package com.imooc.aop.api; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; /** * 引入环绕通知 * * 实现 MethodInterceptor 接口 * @author weijingli * */ public class MoocMethodInterceptor implements MethodInterceptor{ @Override public Object invoke(MethodInvocation methodinvocation) throws Throwable { //实现前置通知的方法 //得到方法 得到类 System.out.println("MoocMethodInterceptor1:"+methodinvocation.getMethod().getName()+" " +methodinvocation.getStaticPart().getClass().getName()); Object obj = methodinvocation.proceed(); //实现后置通知方法 System.out.println("MoocMethodInterceptor2:"+obj); return obj; } }
后置通知:
package com.imooc.aop.api; import java.lang.reflect.Method; import org.springframework.aop.AfterReturningAdvice; /** * 实现AfterReturningAdvice接口,可以在连接点之后执行 * @author weijingli * */ public class MoocAfterReturningAdvice implements AfterReturningAdvice{ /** * obj 作为输出 */ @Override public void afterReturning(Object obj, Method method, Object[] aobj, Object obj1) throws Throwable { System.out.println("MoocAfterReturningAdvice:"+method.getName()+ " 类的名称:"+obj1.getClass().getName()+" "+obj); } }
打印日志:
四月 25, 2019 6:54:59 上午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh 信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Thu Apr 25 06:54:59 CST 2019]; root of context hierarchy 四月 25, 2019 6:54:59 上午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions 信息: Loading XML bean definitions from class path resource [spring-aop-api.xml] MoocBeforeAdvice:sove com.imooc.aop.api.BizLogicImpl MoocMethodInterceptor1:sove java.lang.reflect.Method syso save MoocMethodInterceptor2:BizLogicImpl save MoocAfterReturningAdvice:sove 类的名称:com.imooc.aop.api.BizLogicImpl BizLogicImpl save 四月 25, 2019 6:55:00 上午 org.springframework.context.support.ClassPathXmlApplicationContext doClose 信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Thu Apr 25 06:54:59 CST 2019]; root of context hierarchy