Spring二次代理问题排查

最近遇到一个Spring注入失败的问题。起因是需要在一个Controller ClassB里注入另一个Controller ClassA(一个令人蛋疼的原因),启动的时候报异常,提示需要的类型是ClassA,找到的Bean的实际类型是com.sun.proxy.$Proxy。代码里有一个用来打日志Aop的切面涉及到了ClassA。但是ClassA明明没有实现任何接口啊,proxy对象也应该是使用CGlib生成的,类型应该是ClassA$$EnhancerByCGLIB才对。
把注入Controller的类型改为Object,然后加了断点Debug看了下bean内容。如下图所示,可以看出真正的原始Controller被代理了两次。先是使用CGlib生成的FirstController$$EnhancerBySpringCGLIB类型的代理对象,然后使用JDK动态代理使用这个代理对象做为target又生成了一个com.sun.proxy.$Proxy类型的代理对象

Proxy对象的数据信息

google了一下,看到一篇文章和我遇到的问题很相似:spring二次代理的问题。先说一下问题产生的根本原因:spring上下文里的的BeanPostProcessor列表有多个AbstractAdvisorAutoProxyCreator。spring的动态代理的原理是在bean生命周期的BeanPostProcessor拓展点对bean进行检查是否符合某一个AOP切面的规则,如果需要代理的话,将这个bean替换为代理对象。AbstractAdvisorAutoProxyCreator就是用来生成代理
对象的BeanPostProcessor,它一共有下面四个实现,除了InfrastructureAdvisorAutoProxyCreator,其他的ProxyCreator都能够解析用户通过Advisor类型的Bean定义的切面

DefaultAdvisorAutoProxyCreator //基础的ProxyCreator 
AnnotationAwareAspectJAutoProxyCreator //能够解析AspectJ注解的ProxyCreator
AspectJAwareAdvisorAutoProxyCreator //能够解析AspectJ规则的ProxyCreator
InfrastructureAdvisorAutoProxyCreator //只解析Spring本身基础设施里的切面,忽略用户定义的切面

BeanPostProcessor的机制是从Spring管理的bean中找到所有实现了BeanPostProcessor接口的类,将其注册到beanPostProcessors中,bean会被beanPostProcessors中的每一个Processor处理,也就是说在存在多个ProxyCreator的情况下,bean被之前的ProxyCreator处理以后已经变成了一个代理对象,仍然会被下一个ProxyCreator继续处理,如果这个代理对象仍然符合某一个AOP切面的规则,就会对这个代理对象在产生一个代理对象。

项目中出现问题的代码类似于下面这样,通过Advisor定义了一个切面,规则是所有q.g.controller下的类生效。即使用了aop:aspectj-autoproxy(相当于注册了一个DefaultAdvisorAutoProxyCreator),又注册了一个DefaultAdvisorAutoProxyCreator。这时beanPostProcessors就有了就有了两个ProxyCreator,首先第一个ProxyCreator对controller进行处理,因为controller没有实现任何接口,所以采用CGlib生成了代理对象,代理对象的类型是FirstController$$EnhancerBySpringCGLIB,同时实现了三个接口:org.springframework.aop.SpringProxy
org.springframework.aop.framework.Advised
org.springframework.cglib.proxy.Factory
然后第二个ProxyCreator对这个代理对象进行处理,该代理对象的类的package也是q.g.controller,仍然符合切面的规则。因为该代理对象实现了三个接口,所以就使用JDK基于接口的动态代理生成代理对象。代理对象的类型是$Proxy,package是com.sun.proxy,不再符合切面的规则,即使后面仍然有更多的ProxyCreator,也不会再基于当前的代理对象生成一个新的代理对象

    <aop:aspectj-autoproxy/>
    <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"/>
    <bean id="firstAdvice" class="q.g.aop.FirstAdvice"></bean>
    <bean id="advisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
        <property name="advice">
            <ref bean="firstAdvice" />
        </property>
        <property name="patterns">
            <list>
                <value>q\.g\.controller\..*</value>
            </list>
        </property>
    </bean>

如果将配置改为下面这样,强制使用CGlib生成代理对象,有N个ProxyCreator就会产生N次代理

    <aop:aspectj-autoproxy proxy-target-class="true"/>
    <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator">
        <property name="proxyTargetClass" value="true"/>
    </bean>
    <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator">
        <property name="proxyTargetClass" value="true"/>
    </bean>
    <bean id="firstAdvice" class="q.g.aop.FirstAdvice"></bean>
    <bean id="advisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
        <property name="advice">
            <ref bean="firstAdvice" />
        </property>
        <property name="patterns">
            <list>
                <value>q\.g\.controller\..*</value>
            </list>
        </property>
    </bean>
总结

出现多次代理的条件:

  • 有多个ProxyCreator
  • 生成的代理对象仍然符合切面规则

不要使用多种AOP配置的方式和定义切面时范围尽量精确可以很大程度避免发生这个问题。

另外,杂乱的spring配置文件真是排查这类问题的大敌。不得不吐槽公司项目中的spring 配置文件真的又多又乱,没有一个统一的规则,大家在改动时使用的配置方式、修改的配置文件都不一样,再加上Spring mvc里servlet context 和root context的关系大家一般也不会去关注,很容易埋一些坑在里面。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,744评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,505评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,105评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,242评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,269评论 6 389
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,215评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,096评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,939评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,354评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,573评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,745评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,448评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,048评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,683评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,838评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,776评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,652评论 2 354

推荐阅读更多精彩内容

  • 什么是Spring Spring是一个开源的Java EE开发框架。Spring框架的核心功能可以应用在任何Jav...
    jemmm阅读 16,462评论 1 133
  • 1.什么是Spring框架? Spring是一个轻量级的java开源框架,为了解决企业级应用开发的复杂性创建的ja...
    gskobe0811阅读 519评论 0 3
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,652评论 18 139
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,242评论 11 349
  • 与小人相处的黄金五步法 小人很让人讨厌,比如爱打小报告的人,因为他会把很多你的臭事糟糕事负面事捅到领导那里,让你难...
    创一学习吧阅读 346评论 0 1