最近在项目中遇到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循环引用的处理