spring 提供了多种不同的方案实现对 bean 的 aop proxy, 包括 ProxyFactoryBean, 便利的 TransactionProxyFactoryBean 以及 AutoProxyCreator 等,
下图是 proxy class diagram 以供参考

这里重点说一下最常用的 ProxyFactoryBean, TransactionProxyFactoryBean, BeanNameAutoProxyCreator, DefaultAdvisorAutoProxyCreator 的联系和区别

1.
ProxyFactoryBean : 使用率最高的 proxy 方式, 它通过配置 interceptorNames 属性决定加入哪些
advisor (method interceptor 将会被自动包装成 advisor, 下文将描述这个细节),
注意是 "interceptorNames" 而不是 "interceptors",

原因是 ProxyFactoryBean 可能返回非 singleton 的 proxy 实例, 而 advisior 可能也是非 singleton 的,

因此不能通过 interceptor reference 来注入

2. TransactionProxyFactoryBean : 特定用于 transaction proxy, 注意其 super class 是 AbstractSingletonProxyFactoryBean, 也就是说,

TransactionProxyFactoryBean 永远无法返回非 singleton 的 proxy 实例 !!!

如果你需要非 singleton 的 proxy 实例, 请考虑使用 ProxyFactoryBean.

3.
BeanNameAutoProxyCreator : 故名思义, 根据 bean name 进行 auto proxy, bean name 的
match 规则参见 org.springframework.util.PatternMatchUtils

4. DefaultAdvisorAutoProxyCreator : 更强大的 auto proxy creator, 强大之处在于它会 cahce 容器中所有注册的 advisor, 然后搜索容器中所有的 bean ,
如果某个 bean 满足 advisor 中的 Pointcut, 那么将会被自动代理, 与 BeanNameAutoProxyCreator 相比, 省去了配置 beanNames 的工作,

引用:
eg :

  1. <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" />
  2. <bean id="defaultPointcutAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor" scope="prototype">
  3. <property name="pointcut" ref="fooPointcut"/>
  4. <property name="advice" ref="fooAdvice"/>
  5. </bean>
  6. <bean id="fooAdvice" class="com.mycompany.FooAdvice" scope="prototype" />
  7. <bean id="fooPointcut" class="org.springframework.aop.support.JdkRegexpMethodPointcut" scope="prototype">
  8. <property name="patterns">
  9. <list>
  10. <value>com.mycompany.FooService.*</value>
  11. </list>
  12. </property>
  13. </bean>
  1. <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" />
  2. <bean id="defaultPointcutAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor" scope="prototype">
  3. <property name="pointcut" ref="fooPointcut"/>
  4. <property name="advice" ref="fooAdvice"/>
  5. </bean>
  6. <bean id="fooAdvice" class="com.mycompany.FooAdvice" scope="prototype" />
  7. <bean id="fooPointcut" class="org.springframework.aop.support.JdkRegexpMethodPointcut" scope="prototype">
  8. <property name="patterns">
  9. <list>
  10. <value>com.mycompany.FooService.*</value>
  11. </list>
  12. </property>
  13. </bean>

以上配置将自动代理容器中所有 com.mycompany.FooService 类型的 bean, 并拦截其所有方法

深度话题

1. MethodInterceptor 如何被包装成 Advisor ?

在 AdvisorAdapterRegistry#wrap(Object) 方法中实现, code as below

  1. public Advisor wrap(Object adviceObject) throws UnknownAdviceTypeException {
  2. if (adviceObject instanceof Advisor) {
  3. return (Advisor) adviceObject;
  4. }
  5. if (!(adviceObject instanceof Advice)) {
  6. hrow new UnknownAdviceTypeException(adviceObject);
  7. }
  8. Advice advice = (Advice) adviceObject;
  9. if (advice instanceof MethodInterceptor) {
  10. // So well-known it doesn't even need an adapter.
  11. return new DefaultPointcutAdvisor(advice);
  12. }
  13. ; i < this.adapters.size(); i++) {
  14. / Check that it is supported.
  15. AdvisorAdapter adapter = (AdvisorAdapter) this.adapters.get(i);
  16. if (adapter.supportsAdvice(advice)) {
  17. return new DefaultPointcutAdvisor(advice);
  18. }
  19. }
  20. throw new UnknownAdviceTypeException(advice);
  21. }
  1. public Advisor wrap(Object adviceObject) throws UnknownAdviceTypeException {
  2. if (adviceObject instanceof Advisor) {
  3. return (Advisor) adviceObject;
  4. }
  5. if (!(adviceObject instanceof Advice)) {
  6. hrow new UnknownAdviceTypeException(adviceObject);
  7. }
  8. Advice advice = (Advice) adviceObject;
  9. if (advice instanceof MethodInterceptor) {
  10. // So well-known it doesn't even need an adapter.
  11. return new DefaultPointcutAdvisor(advice);
  12. }
  13. for (int i = 0; i < this.adapters.size(); i++) {
  14. / Check that it is supported.
  15. AdvisorAdapter adapter = (AdvisorAdapter) this.adapters.get(i);
  16. if (adapter.supportsAdvice(advice)) {
  17. return new DefaultPointcutAdvisor(advice);
  18. }
  19. }
  20. throw new UnknownAdviceTypeException(advice);
  21. }

从代码可以看到, 如果 adviceObject(也就是 interceptorNames 对应的 bean) 不是 advisor
而是 MethodInterceptor 或 Advice, 那么 spring 将其包装成 DefaultPointcutAdvisor,
而 DefaultPointcutAdvisor 中定义的 Pointcut 是 TruePointcut :

  1. class TruePointcut implements Pointcut, Serializable {
  2. public static final TruePointcut INSTANCE = new TruePointcut();
  3. /**
  4. * Enforce Singleton pattern.
  5. */
  6. private TruePointcut() {
  7. }
  8. public ClassFilter getClassFilter() {
  9. return ClassFilter.TRUE;
  10. }
  11. public MethodMatcher getMethodMatcher() {
  12. return MethodMatcher.TRUE;
  13. }
  14. /**
  15. * Required to support serialization. Replaces with canonical
  16. * instance on deserialization, protecting Singleton pattern.
  17. * Alternative to overriding <code>equals()</code>.
  18. */
  19. private Object readResolve() {
  20. return INSTANCE;
  21. }
  22. public String toString() {
  23. return "Pointcut.TRUE";
  24. }
  25. }
  1. class TruePointcut implements Pointcut, Serializable {
  2. public static final TruePointcut INSTANCE = new TruePointcut();
  3. /**
  4. * Enforce Singleton pattern.
  5. */
  6. private TruePointcut() {
  7. }
  8. public ClassFilter getClassFilter() {
  9. return ClassFilter.TRUE;
  10. }
  11. public MethodMatcher getMethodMatcher() {
  12. return MethodMatcher.TRUE;
  13. }
  14. /**
  15. * Required to support serialization. Replaces with canonical
  16. * instance on deserialization, protecting Singleton pattern.
  17. * Alternative to overriding <code>equals()</code>.
  18. */
  19. private Object readResolve() {
  20. return INSTANCE;
  21. }
  22. public String toString() {
  23. return "Pointcut.TRUE";
  24. }
  25. }

也就是说, MethodInterceptor 和 Advice 被包装成的 Advisor 将会匹配容器中的所有 bean,
所以, 永
远不要在 DefaultAdvisorAutoProxyCreator 的 interceptorNames 中引用一个 Advice,
那将会使容器中所有的 bean 被自动代理!!! 此时应该考虑使用 BeanNameAutoProxyCreator

Spring AOP 的proxy详解的更多相关文章

  1. Spring框架系列(9) - Spring AOP实现原理详解之AOP切面的实现

    前文,我们分析了Spring IOC的初始化过程和Bean的生命周期等,而Spring AOP也是基于IOC的Bean加载来实现的.本文主要介绍Spring AOP原理解析的切面实现过程(将切面类的所 ...

  2. Spring框架系列(10) - Spring AOP实现原理详解之AOP代理的创建

    上文我们介绍了Spring AOP原理解析的切面实现过程(将切面类的所有切面方法根据使用的注解生成对应Advice,并将Advice连同切入点匹配器和切面类等信息一并封装到Advisor).本文在此基 ...

  3. Spring框架系列(11) - Spring AOP实现原理详解之Cglib代理实现

    我们在前文中已经介绍了SpringAOP的切面实现和创建动态代理的过程,那么动态代理是如何工作的呢?本文主要介绍Cglib动态代理的案例和SpringAOP实现的原理.@pdai Spring框架系列 ...

  4. Spring框架系列(12) - Spring AOP实现原理详解之JDK代理实现

    上文我们学习了SpringAOP Cglib动态代理的实现,本文主要是SpringAOP JDK动态代理的案例和实现部分.@pdai Spring框架系列(12) - Spring AOP实现原理详解 ...

  5. Spring Aop底层原理详解

    Spring Aop底层原理详解(来源于csdn:https://blog.csdn.net/baomw)

  6. Spring AOP 入门实例详解

    目录 AOP概念 AOP核心概念 Spring对AOP的支持 基于Spring的AOP简单实现 基于Spring的AOP使用其他细节 AOP概念 AOP(Aspect Oriented Program ...

  7. Spring框架系列(8) - Spring IOC实现原理详解之Bean实例化(生命周期,循环依赖等)

    上文,我们看了IOC设计要点和设计结构:以及Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的:容器中存放的是Bean的定义即Be ...

  8. Spring框架系列(6) - Spring IOC实现原理详解之IOC体系结构设计

    在对IoC有了初步的认知后,我们开始对IOC的实现原理进行深入理解.本文将帮助你站在设计者的角度去看IOC最顶层的结构设计.@pdai Spring框架系列(6) - Spring IOC实现原理详解 ...

  9. Spring框架系列(7) - Spring IOC实现原理详解之IOC初始化流程

    上文,我们看了IOC设计要点和设计结构:紧接着这篇,我们可以看下源码的实现了:Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的. ...

随机推荐

  1. R语言学习笔记︱Echarts与R的可视化包——地区地图

    笔者寄语:感谢CDA DSC训练营周末上完课,常老师.曾柯老师加了小课,讲了echart与R结合的函数包recharts的一些基本用法.通过对比谢益辉老师GitHub的说明文档,曾柯老师极大地简化了一 ...

  2. 查看dmp文件

    1.查看dmp文件,首先要通过以下的链接地址进行下载 http://msdl.microsoft.com/download/symbols/debuggers/dbg_x86_6.11.1.404.m ...

  3. 图像处理------泛洪填充算法(Flood Fill Algorithm) 油漆桶功能

    泛洪填充算法(Flood Fill Algorithm) 泛洪填充算法又称洪水填充算法是在很多图形绘制软件中常用的填充算法,最熟悉不过就是 windows paint的油漆桶功能.算法的原理很简单,就 ...

  4. 图像处理------透明混合 - Alpha Blending效果

    基本原理: 图像的透明混合有个专属名词– Alpha Blending 对任意两张图像可以合成为一张图像,合成图像的像素取值根据数学公式: RGB3 = (1- a) * RGB1 + a * RGB ...

  5. 错误代码: 1052 Column 'stu_id' in field list is ambiguous

    1.错误描述 1 queries executed, 0 success, 1 errors, 0 warnings 查询:select stu_id, (SELECT stu_name FROM t ...

  6. 页面某些特定图标的权限,比如导入导出表格,下载等等,这个权限必须在有某个页面查看的权利的基础上(细粒度)(shiro项目中来的四)

    一,查找按钮权限的设置 第一步:会根据用户的相关信息去查到它的角色表: SELECT * FROM SYS_USER WHERE user_id='eded77bdf35347249b2bacfa18 ...

  7. 使用pyh生成HTML文档

    title: 使用pyh生成HTML文档 tags: [python3, 爬虫,pyh] date: 2018-03-09 21:01:34 categories: Python keywords: ...

  8. 移动端的silder,未封装,基于zepto的touch模块,有参照修改过touch的bug

    <!--html模块--> <header class="appoin-head"> <ul> <li class="aa&qu ...

  9. 【HNOI2012】永无乡(splay,启发式合并)

    题解 Description 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过 ...

  10. [USACO07NOV]Cow Relays

    map+floyed+矩阵乘法(倍增floyed) # include <stdio.h> # include <stdlib.h> # include <iostrea ...