2

如果我确实在一个方法上使用了事务注释,同时也使用了 Aspect,那么 spring 将如何表现呢?它会在事务代理对象上创建方面代理吗?或者 spring 是否聪明地混淆了代理对象的逻辑?

如果我的理解在这里完全错误,请纠正我。

4

1 回答 1

2

AOP 代理是由BeanPostProcessor一个在hierarchy 中最具体的一个AbstractAutoProxyCreator通过以下步骤创建的

  • 查找可以应用于 bean 的建议,请参阅AopUtils.findAdvisorsThatCanApply()
  • 使用 对顾问进行排序OrderComparator,请参阅AbstractAdvisorAutoProxyCreator.sortAdvisors()
  • 使用顾问创建代理。

所以通常只涉及一个代理。

然而,正如 Marten 所说,如果您通过 AutorProxyCreator 未知的其他方式创建代理,您可以轻松获得代理的代理。

例如:

<bean id="proxy" class="org.springframework.aop.framework.ProxyFactoryBean">
    <property name="target" ref="target" />
    <property name="proxyTargetClass" value="true" />
    <property name="interceptorNames" value="tracer" />
</bean> 

<bean id="target" class="test.SomeBean" />

<bean id="tracer" class="test.Tracer" />

<aop:config proxy-target-class="true">
    <aop:advisor id="traceAdvisor" advice-ref="tracer" pointcut="execution (public * *(..))" />
</aop:config>

public class SomeBean {

    public void someMethod() {
        System.out.println("In someMethod");
    }

    public static void main(String[] args) {
        ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("/context.xml");
        SomeBean bean = (SomeBean) ctx.getBean("proxy");
        bean.someMethod();

    }
}

public class Tracer  implements MethodBeforeAdvice {

    @Override
    public void before(Method method, Object[] args, Object target)
            throws Throwable {
        System.out.println("About to execute [" + method.getName() + "]" +
            " on target [" + target.getClass().getName() + "]");

    }

}

将输出:

About to execute [someMethod] on target [test.SomeBean$$EnhancerByCGLIB$$428125af]
About to execute [someMethod] on target [test.SomeBean$$EnhancerByCGLIB$$ee348b75]
About to execute [someMethod] on target [test.SomeBean]
In someMethod
于 2014-01-09T19:43:08.660 回答