Aop相关阅读
阅读本文之前,需要先掌握下面3篇文章内容,不然会比较吃力。
本文继续Aop。
AOP创建代理的方式主要分为2大类
手动方式
也称为手动的方式,需要通过硬编码一个个创建代理。
自动化的方式
也称为批量的方式,批量的方式用在spring环境中,通过bean后置处理器来对符合条件的bean创建代理
手动的方式基本上是采用硬编码的方式,相对来说更灵活一些,可以脱离spring环境使用,而自动化的方式主要用在spring环境中,和spring集成起来更容易一些,更强大一些。
AOP创建代理相关的类

左边的ProxyCreatorSupport下面的都是手动的方式,有3个类。
右边的AbstractAutoProxyCreator下面挂的都是自动创建代理的方式,主要有5个实现类。
手动3种方式

ProxyFactory方式
这种是硬编码的方式,可以脱离spring直接使用,用到的比较多,自动化方式创建代理中都是依靠ProxyFactory来实现的,所以这种方式的原理大家一定要了解,上篇文章中已经有介绍过了,不清楚的可以去看一下:Spring系列第32篇:AOP核心源码、原理详解
AspectJProxyFactory方式
AspectJ是一个面向切面的框架,是目前最好用,最方便的AOP框架,Spring将其集成进来了,通过Aspectj提供的一些功能实现aop代理非常方便,下篇文章将详解。
ProxyFactoryBean方式
Spring环境中给指定的bean创建代理的一种方式,本文主要介绍这个。
ProxyFactoryBean
这个类实现了一个接口FactoryBean,FactoryBean不清楚的可以看一下:Spring系列第5篇:创建bean实例这些方式你们都知道?
ProxyFactoryBean就是通过FactoryBean的方式来给指定的bean创建一个代理对象。
创建代理,有3个信息比较关键:
- 需要增强的功能,这个放在通知(Advice)中实现
- 目标对象(target):表示你需要给哪个对象进行增强
- 代理对象(proxy):将增强的功能和目标对象组合在一起,然后形成的一个代理对象,通过代理对象来访问目标对象,起到对目标对象增强的效果。
使用ProxyFactoryBean也是围绕着3部分来的,ProxyFactoryBean使用的步骤:
1.创建ProxyFactoryBean对象2.通过ProxyFactoryBean.setTargetName设置目标对象的bean名称,目标对象是spring容器中的一个bean3.通过ProxyFactoryBean。setInterceptorNames添加需要增强的通知4.将ProxyFactoryBean注册到Spring容器,假设名称为proxyBean5.从Spring查找名称为proxyBean的bean,这个bean就是生成好的代理对象
上案例。
来个类Service1
package com.javacode2018.aop.demo8.test1;public class Service1 {public void m1() {System.out.println("我是 m1 方法");}public void m2() {System.out.println("我是 m2 方法");}}
需求
在spring容器中注册上面这个类的bean,名称为
service1,通过代理的方式来对这个bean进行增强,来2个通知一个前置通知:在调用
service1中的任意方法之前,输出一条信息:准备调用xxxx方法一个环绕通知:复制统计所有方法的耗时。
下面是代码的实现
package com.javacode2018.aop.demo8.test1;import org.aopalliance.intercept.MethodInterceptor;import org.aopalliance.intercept.MethodInvocation;import org.springframework.aop.MethodBeforeAdvice;import org.springframework.aop.framework.ProxyFactoryBean;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.lang.Nullable;import java.lang.reflect.Method;@Configurationpublic class MainConfig1 {//注册目标对象@Beanpublic Service1 service1() {return new Service1();}//注册一个前置通知@Beanpublic MethodBeforeAdvice beforeAdvice() {MethodBeforeAdvice advice = new MethodBeforeAdvice() {@Overridepublic void before(Method method, Object[] args, @Nullable Object target) throws Throwable {System.out.println("准备调用:" + method);}};return advice;}//注册一个后置通知@Beanpublic MethodInterceptor costTimeInterceptor() {MethodInterceptor methodInterceptor = new MethodInterceptor() {@Overridepublic Object invoke(MethodInvocation invocation) throws Throwable {long starTime = System.nanoTime();Object result = invocation.proceed();long endTime = System.nanoTime();System.out.println(invocation.getMethod() + ",耗时(纳秒):" + (endTime - starTime));return result;}};return methodInterceptor;}//注册ProxyFactoryBean@Beanpublic ProxyFactoryBean service1Proxy() {//1.创建ProxyFactoryBeanProxyFactoryBean proxyFactoryBean = new ProxyFactoryBean();//2.设置目标对象的bean名称proxyFactoryBean.setTargetName("service1");//3.设置拦截器的bean名称列表,此处2个(advice1和advice2)proxyFactoryBean.setInterceptorNames("beforeAdvice", "costTimeInterceptor");return proxyFactoryBean;}}
下面启动spring容器,并获取代理对象
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MainConfig1.class);//获取代理对象,代理对象bean的名称为注册ProxyFactoryBean的名称,即:service1ProxyService1 bean = context.getBean("service1Proxy", Service1.class);System.out.println("----------------------");//调用代理的方法bean.m1();System.out.println("----------------------");//调用代理的方法bean.m2();
运行输出
----------------------准备调用:public void com.javacode2018.aop.demo8.test1.Service1.m1()我是 m1 方法public void com.javacode2018.aop.demo8.test1.Service1.m1(),耗时(纳秒):8680400----------------------准备调用:public void com.javacode2018.aop.demo8.test1.Service1.m2()我是 m2 方法public void com.javacode2018.aop.demo8.test1.Service1.m2(),耗时(纳秒):82400
从输出中可以看到,目标对象service1已经被增强了。
ProxyFactoryBean中的interceptorNames
interceptorNames用来指定拦截器的bean名称列表,常用的2种方式。
- 批量的方式
- 非批量的方式
批量的方式
使用方法
proxyFactoryBean.setInterceptorNames("需要匹配的bean名称*");
需要匹配的bean名称后面跟上一个*,可以用来批量的匹配,如:interceptor*,此时spring会从容器中找到下面2中类型的所有bean,bean名称以interceptor开头的将作为增强器
org.springframework.aop.Advisororg.aopalliance.intercept.Interceptor
这个地方使用的时候需要注意,批量的方式注册的时候,如果增强器的类型不是上面2种类型的,比如下面3种类型通知,我们需要将其包装为Advisor才可以,而MethodInterceptor是Interceptor类型的,可以不用包装为Advisor类型的。
MethodBeforeAdvice(方法前置通知)AfterReturningAdvice(方法后置通知)ThrowsAdvice(异常通知)
下面来个案例感受一下。
案例
下面批量注册2个增强器。
package com.javacode2018.aop.demo8.test2;import com.javacode2018.aop.demo8.test1.Service1;import org.aopalliance.intercept.MethodInterceptor;import org.aopalliance.intercept.MethodInvocation;import org.springframework.aop.Advisor;import org.springframework.aop.MethodBeforeAdvice;import org.springframework.aop.framework.ProxyFactoryBean;import org.springframework.aop.support.DefaultPointcutAdvisor;import org.springframework.context.annotation.Bean;import org.springframework.lang.Nullable;import java.lang.reflect.Method;public class MainConfig2 {//注册目标对象@Beanpublic Service1 service1() {return new Service1();}//定义一个增强器:interceptor1,内部是一个前置通知,需要将其包装为Advisor类型的@Beanpublic Advisor interceptor1() {MethodBeforeAdvice advice = new MethodBeforeAdvice() {@Overridepublic void before(Method method, Object[] args, @Nullable Object target) throws Throwable {System.out.println("准备调用:" + method);}};DefaultPointcutAdvisor advisor = new DefaultPointcutAdvisor();advisor.setAdvice(advice);return advisor;}//定义一个增强器:interceptor2@Beanpublic MethodInterceptor interceptor2() {MethodInterceptor methodInterceptor = new MethodInterceptor() {@Overridepublic Object invoke(MethodInvocation invocation) throws Throwable {long starTime = System.nanoTime();Object result = invocation.proceed();long endTime = System.nanoTime();System.out.println(invocation.getMethod() + ",耗时(纳秒):" + (endTime - starTime));return result;}};return methodInterceptor;}//注册ProxyFactoryBean@Beanpublic ProxyFactoryBean service1Proxy() {//1.创建ProxyFactoryBeanProxyFactoryBean proxyFactoryBean = new ProxyFactoryBean();//2.设置目标对象的bean名称proxyFactoryBean.setTargetName("service1");//3.设置拦截器的bean名称列表,此处批量注册proxyFactoryBean.setInterceptorNames("interceptor*"); //@1return proxyFactoryBean;}}
上面定义了2个增强器:
interceptor1:前置通知,包装为Advisor类型了
interceptor2:环绕通知,MethodInterceptor类型的
测试代码
@Testpublic void test2() {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MainConfig2.class);//获取代理对象,代理对象bean的名称为注册ProxyFactoryBean的名称,即:service1ProxyService1 bean = context.getBean("service1Proxy", Service1.class);System.out.println("----------------------");//调用代理的方法bean.m1();System.out.println("----------------------");//调用代理的方法bean.m2();}
运行输出
----------------------准备调用:public void com.javacode2018.aop.demo8.test1.Service1.m1()我是 m1 方法public void com.javacode2018.aop.demo8.test1.Service1.m1(),耗时(纳秒):10326200----------------------准备调用:public void com.javacode2018.aop.demo8.test1.Service1.m2()我是 m2 方法public void com.javacode2018.aop.demo8.test1.Service1.m2(),耗时(纳秒):52000
非批量的方式
用法
非批量的方式,需要注册多个增强器,需明确的指定多个增强器的bean名称,多个增强器按照参数中指定的顺序执行,如
proxyFactoryBean.setInterceptorNames("advice1","advice2");
advice1、advice2对应的bean类型必须为下面列表中指定的类型,类型这块比匹配的方式范围广一些
MethodBeforeAdvice(方法前置通知)AfterReturningAdvice(方法后置通知)ThrowsAdvice(异常通知)org.aopalliance.intercept.MethodInterceptor(环绕通知)org.springframework.aop.Advisor(顾问)
下面来个案例。
案例
这次给service1来3个通知:前置、环绕、后置
package com.javacode2018.aop.demo8.test3;import com.javacode2018.aop.demo8.test1.Service1;import org.aopalliance.intercept.MethodInterceptor;import org.aopalliance.intercept.MethodInvocation;import org.springframework.aop.AfterReturningAdvice;import org.springframework.aop.MethodBeforeAdvice;import org.springframework.aop.framework.ProxyFactoryBean;import org.springframework.context.annotation.Bean;import org.springframework.lang.Nullable;import java.lang.reflect.Method;public class MainConfig3 {//注册目标对象@Beanpublic Service1 service1() {return new Service1();}//定义一个前置通知@Beanpublic MethodBeforeAdvice methodBeforeAdvice() {MethodBeforeAdvice methodBeforeAdvice = new MethodBeforeAdvice() {@Overridepublic void before(Method method, Object[] args, @Nullable Object target) throws Throwable {System.out.println("准备调用:" + method);}};return methodBeforeAdvice;}//定义一个环绕通知@Beanpublic MethodInterceptor methodInterceptor() {MethodInterceptor methodInterceptor = new MethodInterceptor() {@Overridepublic Object invoke(MethodInvocation invocation) throws Throwable {long starTime = System.nanoTime();Object result = invocation.proceed();long endTime = System.nanoTime();System.out.println(invocation.getMethod() + ",耗时(纳秒):" + (endTime - starTime));return result;}};return methodInterceptor;}//定义一个后置通知@Beanpublic AfterReturningAdvice afterReturningAdvice() {AfterReturningAdvice afterReturningAdvice = new AfterReturningAdvice() {@Overridepublic void afterReturning(@Nullable Object returnValue, Method method, Object[] args, @Nullable Object target) throws Throwable {System.out.println(method + ",执行完毕!");}};return afterReturningAdvice;}//注册ProxyFactoryBean@Beanpublic ProxyFactoryBean service1Proxy() {//1.创建ProxyFactoryBeanProxyFactoryBean proxyFactoryBean = new ProxyFactoryBean();//2.设置目标对象的bean名称proxyFactoryBean.setTargetName("service1");//3.设置拦截器的bean名称列表,此处批量注册proxyFactoryBean.setInterceptorNames("methodBeforeAdvice", "methodInterceptor", "afterReturningAdvice");return proxyFactoryBean;}}
测试代码
@Testpublic void test3() {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MainConfig3.class);//获取代理对象,代理对象bean的名称为注册ProxyFactoryBean的名称,即:service1ProxyService1 bean = context.getBean("service1Proxy", Service1.class);System.out.println("----------------------");//调用代理的方法bean.m1();System.out.println("----------------------");//调用代理的方法bean.m2();}
运行输出
----------------------准备调用:public void com.javacode2018.aop.demo8.test1.Service1.m1()我是 m1 方法public void com.javacode2018.aop.demo8.test1.Service1.m1(),执行完毕!public void com.javacode2018.aop.demo8.test1.Service1.m1(),耗时(纳秒):12724100----------------------准备调用:public void com.javacode2018.aop.demo8.test1.Service1.m2()我是 m2 方法public void com.javacode2018.aop.demo8.test1.Service1.m2(),执行完毕!public void com.javacode2018.aop.demo8.test1.Service1.m2(),耗时(纳秒):76700
源码解析
重点在于下面这个方法
org.springframework.aop.framework.ProxyFactoryBean#getObject
源码:
public Object getObject() throws BeansException {//初始化advisor(拦截器)链initializeAdvisorChain();//是否是单例if (isSingleton()) {//创建单例代理对象return getSingletonInstance();}else {//创建多例代理对象return newPrototypeInstance();}}
initializeAdvisorChain方法,用来初始化advisor(拦截器)链,是根据interceptorNames配置,找到spring容器中符合条件的拦截器,将其放入创建aop代理的配置中
private synchronized void initializeAdvisorChain() throws AopConfigException, BeansException {if (!ObjectUtils.isEmpty(this.interceptorNames)) {// 轮询 interceptorNamesfor (String name : this.interceptorNames) {//批量注册的方式:判断name是否以*结尾if (name.endsWith(GLOBAL_SUFFIX)) {//@1:从容器中匹配查找匹配的增强器,将其添加到aop配置中addGlobalAdvisor((ListableBeanFactory) this.beanFactory,name.substring(0, name.length() - GLOBAL_SUFFIX.length()));}else {//非匹配的方式:按照name查找bean,将其包装为Advisor丢到aop配置中Object advice;//从容器中查找beanadvice = this.beanFactory.getBean(name);//@2:将advice添加到拦截器列表中addAdvisorOnChainCreation(advice, name);}}}}
@1:addGlobalAdvisor批量的方式Advisor,看一下源码,比较简单
/*** 添加所有全局拦截器和切入点,* 容器中所有类型为Advisor/Interceptor的bean,bean名称prefix开头的都会将其添加到拦截器链中*/private void addGlobalAdvisor(ListableBeanFactory beanFactory, String prefix) {//获取容器中所有类型为Advisor的beanString[] globalAdvisorNames =BeanFactoryUtils.beanNamesForTypeIncludingAncestors(beanFactory, Advisor.class);//获取容器中所有类型为Interceptor的beanString[] globalInterceptorNames =BeanFactoryUtils.beanNamesForTypeIncludingAncestors(beanFactory, Interceptor.class);List<Object> beans = new ArrayList<>(globalAdvisorNames.length + globalInterceptorNames.length);Map<Object, String> names = new HashMap<>(beans.size());for (String name : globalAdvisorNames) {Object bean = beanFactory.getBean(name);beans.add(bean);names.put(bean, name);}for (String name : globalInterceptorNames) {Object bean = beanFactory.getBean(name);beans.add(bean);names.put(bean, name);}//对beans进行排序,可以实现Ordered接口,排序规则:order ascAnnotationAwareOrderComparator.sort(beans);for (Object bean : beans) {String name = names.get(bean);//判断bean是否已prefix开头if (name.startsWith(prefix)) {//将其添加到拦截器链中addAdvisorOnChainCreation(bean, name);}}}
@2:addAdvisorOnChainCreation
private void addAdvisorOnChainCreation(Object next, String name) {//namedBeanToAdvisor用来将bean转换为advisorAdvisor advisor = namedBeanToAdvisor(next);//将advisor添加到拦截器链中addAdvisor(advisor);}
namedBeanToAdvisor方法
private AdvisorAdapterRegistry advisorAdapterRegistry = new DefaultAdvisorAdapterRegistry();private Advisor namedBeanToAdvisor(Object next) {//将对象包装为Advisor对象return this.advisorAdapterRegistry.wrap(next);}
对AdvisorAdapterRegistry不清楚的,看一下上一篇文章:Spring系列第32篇:AOP核心源码、原理详解
advisorAdapterRegistry#wrap方法将adviceObject包装为Advisor对象,代码如下,比较简单
public Advisor wrap(Object adviceObject) throws UnknownAdviceTypeException {if (adviceObject instanceof Advisor) {return (Advisor) adviceObject;}if (!(adviceObject instanceof Advice)) {throw new UnknownAdviceTypeException(adviceObject);}Advice advice = (Advice) adviceObject;if (advice instanceof MethodInterceptor) {return new DefaultPointcutAdvisor(advice);}for (AdvisorAdapter adapter : this.adapters) {if (adapter.supportsAdvice(advice)) {return new DefaultPointcutAdvisor(advice);}}throw new UnknownAdviceTypeException(advice);}
总结
- spring中创建代理主要分为2种:手动方式和自动化的方式
- 手动方式采用硬编码的方式,一次只能给一个目标对象创建代理对象,相对来说灵活一下,对开发者来说更灵活一些,通常可以独立spring环境使用;自动化的方式主要在spring环境中使用,通常是匹配的方式来为符合条件的目标bean创建代理,使用起来更简单一些
- 本文介绍的
ProxyFactoryBean用来在spring环境中给指定的bean创建代理对象,用到的不是太多,大家可以作为了解即可
最新资料
注意:本文归作者所有,未经作者允许,不得转载