美文网首页技术
springAOP应尽量避免自己创建AutoProxyCreat

springAOP应尽量避免自己创建AutoProxyCreat

作者: carway | 来源:发表于2017-12-24 23:19 被阅读1623次

其实这个话题是紧接着上一篇文章的。

再接着讲spring AOP API 的坑

出问题的配置

//配置1
<bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">
  <property name="proxyTargetClass" value="true"/>
</bean>
<tx:annotation-driven transaction-manager="transactionManager"/>

此配置的目的是想进行cglib类代理。但是实际上当进行直接注入类,而不是接口时会找不到Bean错误。

但是如果是这样配置:
//配置2
<aop:aspectj-autoproxy proxy-target-class="true"/>
<tx:annotation-driven transaction-manager="transactionManager"/>

此配置可以很好的工作,并注入类(不是接口)。

分析

1、<aop:aspectj-autoproxy proxy-target-class="true"> 该命名空间会交给org.springframework.aop.config.AopNamespaceHandler处理:

registerBeanDefinitionParser("aspectj-autoproxy", new AspectJAutoProxyBeanDefinitionParser());

在AspectJAutoProxyBeanDefinitionParser中,会执行parse方法解析配置:

public BeanDefinition parse(Element element, ParserContext parserContext) {
    AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);
    extendBeanDefinition(element, parserContext);
    return null;
}

其中AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);目的是注册AnnotationAwareAspectJAutoProxyCreator:

return registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source);

但是注意了:

if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {
  BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);
  if (!cls.getName().equals(apcDefinition.getBeanClassName())) {
    int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());
    int requiredPriority = findPriorityForClass(cls);
    if (currentPriority < requiredPriority) {
      apcDefinition.setBeanClassName(cls.getName());
    }
  }
  return null;
}

大家可以看到一句话:

  • if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME))
  • AUTO_PROXY_CREATOR_BEAN_NAME=“org.springframework.aop.config.internalAutoProxyCreator”,
  • 即首先判断当前容器中是否包含名字为AUTO_PROXY_CREATOR_BEAN_NAME的Bean, 如果包含:然后判断优先级,谁优先级高谁获胜,即最后那个获胜的是实际的AutoProxyCreator

到此我们可以看到跟"<bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">"配置没什么区别,除了没有名字外。

2、接下来看一下<tx:annotation-driven>:

该命名空间交给org.springframework.transaction.config.TxNamespaceHandler处理:

registerBeanDefinitionParser("annotation-driven", new AnnotationDrivenBeanDefinitionParser());

其中<annotation-driven> 会交给AnnotationDrivenBeanDefinitionParser进行解析:

public BeanDefinition parse(Element element, ParserContext parserContext) {
  String mode = element.getAttribute("mode");
  if ("aspectj".equals(mode)) {
    // mode="aspectj"
    registerTransactionAspect(element, parserContext);
  }else {
    // mode="proxy"
    AopAutoProxyConfigurer.configureAutoProxyCreator(element, parserContext);
  }
  return null;
}

默认mode="proxy",所以走AopAutoProxyConfigurer.configureAutoProxyCreator,其代码中第一句话是:

AopNamespaceUtils.registerAutoProxyCreatorIfNecessary(parserContext, element);
public static void registerAutoProxyCreatorIfNecessary(
                  ParserContext parserContext, Element sourceElement) {

  BeanDefinition beanDefinition = AopConfigUtils.registerAutoProxyCreatorIfNecessary(
    parserContext.getRegistry(), parserContext.extractSource(sourceElement));
    useClassProxyingIfNecessary(parserContext.getRegistry(), sourceElement);
    registerComponentIfNecessary(beanDefinition, parserContext);
 }

AopConfigUtils.registerAutoProxyCreatorIfNecessary是:

registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);
private static BeanDefinition registerOrEscalateApcAsRequired(Class cls, BeanDefinitionRegistry registry, Object source) {
  Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
  if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {
    BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);
    if (!cls.getName().equals(apcDefinition.getBeanClassName())) {
      int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());
      int requiredPriority = findPriorityForClass(cls);
      if (currentPriority < requiredPriority) {
        apcDefinition.setBeanClassName(cls.getName());
      }
    }
    return null;
  }
//省略

}

此处我们又看到了registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME),如果是:

  • 配置1,那么实际是两个AutoProxyCreator。前面<bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">定义了一个AutoProxyCreator,后面的<tx:annotation-driven transaction-manager="transactionManager"/>也定义另一个AutoProxyCreator。
  • 配置2,那么实际是共用一个AutoProxyCreator;

而且如果配置1时,因为我们没有指定<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/> 所以是JDK动态代理,因此不管怎么样,都无法注入类的。

问题找到了,原因是注册了两个AutoProxyCreator,造成了二次代理引发的问题,这个和之前的《spring的bean二次代理问题》一样

如何解决

  • 给配置1起名字为”org.springframework.aop.config.internalAutoProxyCreator“;
  • 或者使用配置2

建议

1、如果没有必要,请不要使用低级别API,如上述-->自己去创建AutoProxyCreator
2、首先选择使用如:
<aop:config>或者是<org.springframework.aop.config.internalAutoProxyCreator>

如上配置已经非常好了,根本没必要使用低级别API。
如<tx:annotation-driven>使用的AutoProxyCreator都是和上边是一样的。这样还能防止二次代理。

声明式/@AspectJ风格的AOP都非常好了,完全没必要使用低级别API,请不要再使用低级别API了。

如何用过shiro的朋友都应该知道如下配置:

<!-- Enable Shiro Annotations for Spring-configured beans.  Only run after -->
<!-- the lifecycleBeanProcessor has run: -->
<bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" depends-on="lifecycleBeanPostProcessor"/>
<bean class="org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor">
    <property name="securityManager" ref="securityManager"/>
</bean>

其实我们可以换成这样:

<aop:config proxy-target-class="true"/>
<bean class="org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor">
  <property name="securityManager" ref="securityManager"/>
</bean>

或者使用<aop:aspectj-autoproxy>也行,这样也不会存在二次代理的问题。

至于shiro这个之前的配置容易发生二次注入,我会再下一篇文章《在springmvc+shiro项目中使用druid监听controller层出现IllegalStateException异常》详细讲到

大部分内容参考于http://jinnianshilongnian.iteye.com/blog/1901694

相关文章

  • springAOP应尽量避免自己创建AutoProxyCreat

    其实这个话题是紧接着上一篇文章的。 再接着讲spring AOP API 的坑 出问题的配置 此配置的目的是想进行...

  • Android 如何提高代码质量?

    避免创建不必要的对象,尽可能避免频繁的创建临时对象,例如在for循环内,减少GC的次数。 尽量使用基本数据类型代替...

  • order by 关键字的排序优化

    ORDER BY 子句,尽量使用index方式排序,避免使用FileSort方式排序。 建表 插入数据 创建索引 ...

  • 疫情当下,中兴新支点多链路聚合设备解决远程办公网络问题

    疫情当前,应政府要求,尽量避免外出,大多数企业安排员工在家进行远程办公,避免人群聚集的办公场所。 受疫情影响,许多...

  • 编写内聚的代码【文摘】

    在程序开发过程中,我们应该让一个类的功能尽量集中,让组件尽量小。 要避免创建很大的类或者组件,也不要创建无所不包的...

  • 时尚搭配

    颜色:在挑选衣服的时候,尽量以适合自己肤色的衣服为主。例如,肤色稍稍有些黑的朋友,应避免挑选黑色系的衣服。可以挑选...

  • SpringAOP

    springAOP实现方式 1.动态代理 使用 Proxy机制,使用反射技术,创建实际对象的代理对象,添加AOP的...

  • 2022-07-08

    2022.7.8日精进 今日体验。 不管是谁的错误尽量避免点,尽量避免损失,避免耽误时间,

  • Spring笔记04_AOP注解开发_模板_事务

    1. Spring基于AspectJ的注解的AOP开发 1. 1 SpringAOP的注解入门 创建项目,导入ja...

  • Mysql查询优化

    1、对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引。2、应...

网友评论

    本文标题:springAOP应尽量避免自己创建AutoProxyCreat

    本文链接:https://www.haomeiwen.com/subject/ulhkgxtx.html