入门客AI创业平台(我带你入门,你带我飞行)
博文笔记

Spring中bean循环引用的处理

创建时间:2016-10-09 投稿人: 浏览次数:2300

最近在项目中遇到Bean循环引用的问题
问题是这样的:

Caused by: org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name "storageQueryService": Bean with name "storageQueryService" has been injected into other beans [storageModifyService] in its raw version as part of a circular reference, but has eventually been wrapped. This means that said other beans do not use the final version of the bean. This is often the result of over-eager type matching - consider using "getBeanNamesOfType" with the "allowEagerInit" flag turned off, for example.
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:569)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:476)
    at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:302)
    at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:229)
    at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:298)
    at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:193)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.findAutowireCandidates(DefaultListableBeanFactory.java:1081)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:1006)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:904)
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:527)
    ... 51 more

场景是这样的:

  • 文件服务有读写服务,各自有独立的业务逻辑
  • 读服务需要依赖写服务(第一次访问根节点时需要把根节点初始化出)
  • 写服务需要依赖读服务(需要把操作记录查出才可以操作)

由于用户体系是被动响应(被动同步),而且分区是基于业务的,并不是用户操作,所以无法准确的得到初次访问的请求来初始化根节点,而且可能产生的并发问题也需要有策略兜底,保证请求可以成功。

在这个场景下,如果从设计层面拆分,底层解决会有重复代码,顶层解决会有效率问题(RPC服务接口),无奈想办法解决Spring中Bean循环引用的问题。

在一般情况下,这个配置在现有的spring3.0中是可以正常工作的,前提是没有对beanA和beanB进行增强(代理)。但是,如果任意一方进行了增强,比如通过spring的代理对beanA进行了增强,即实际返回的对象和原始对象不一致的情况,在这种情况下,就会报上述错误。

这个错误即对于一个bean,其所引用的对象并不是由spring容器最终生成的对象,而只是一个原始对象,而spring不允许这种情况出现,即持有过程中间对象。

那么,如何处理这种循环引用呢?答案其实也很简单,将两方的循环切掉。然后使用一个BeanPostProcessor即可。

@Component
public class BeanPostProcessor {
    @Autowired
    StorageModifyService modifyService;
    @Autowired
    StorageQueryService queryService;

    @PostConstruct
    public void init() {
        modifyService.setQueryService(queryService);
        queryService.setStorageModifyService(modifyService);
    }

}

参考链接:Spring中Bean循环引用的处理

声明:该文观点仅代表作者本人,入门客AI创业平台信息发布平台仅提供信息存储空间服务,如有疑问请联系rumenke@qq.com。
  • 上一篇:没有了
  • 下一篇:没有了
未上传头像