文章目录
- 【项目经验】记一次feign注入报循环依赖错误解决方法
- 1、环境报错
- 2、问题分析
- 3、解决方法(两种方法均可)
- 3.1 直接手动注入
- ApplicationContext的getBean方法来获取Spring容器中已初始化的bean。getBean一共有以下四种方法原型:
- getBean(String name)
- getBean(Class type)
- getBean(String name,Class type)
- getBean(String name,Object[] args)
- 1. getBean(String name)
- 2. getBean(Class type)
- 相同点:都要求id或者name或者类型在容器中的唯一性。
- 不同点:getBean(String name)获得的对象需要类型转换而getBean(Class type)获得的对象无需类型转换。
- 3. getBean(String name,Class type)
- 4、getBean(String name,Object[] args)
- 3.2 延迟加载
【项目经验】记一次feign注入报循环依赖错误解决方法
1、环境报错
Caused by: org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'mvcResourceUrlProvider': Requested bean is currently in creation: Is there an unresolvable circular reference?
2、问题分析
1、排查feign循环依赖的类属于哪个jar包
2、查看pom文件,是否具有循环依赖
3、mvcResourceUrlProvider发生了循环依赖,经过排查后发现是所有报错的注入都和两个在拦截器内注入的service有依赖关系,且最下一级都定位到了feignClient,故判定是feign在拦截器内的注入发生了循环依赖
3、解决方法(两种方法均可)
3.1 直接手动注入
FeignUserService feignUserService = SpringContextUtil.getBean(FeignUserService.class);
3.1.1 getBean方法详解
ApplicationContext的getBean方法来获取Spring容器中已初始化的bean。getBean一共有以下四种方法原型:
1. getBean(String name)
参数name表示IOC容器中已经实例化的bean的id或者name,且无论是id还是name都要求在IOC容器中是唯一的不能重名。那么这种方法就是通过id或name去查找获取bean.获取bean的参考代码如下:
@Test public void testPerson() { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); Person p = (Person) ctx.getBean("p"); System.out.println(p); }
2. getBean(Class type)
参数Class type表示要加载的Bean的类型。如果该类型没有继承任何父类(Object类除外)和实现接口的话,那么要求该类型的bean在IOC容器中也必须是唯一的。比如applicationContext.xml配置两个类型完全一致的bean,且都没有配置id和name属性。
<bean class="com.bean.Person"> <property name="name" value="张三"/> <property name="age" value="18"/> </bean> <bean class="com.bean.Person"> <property name="name" value="李四"/> <property name="age" value="20"/> </bean>
那么通过com.bean.Person这种类型来查找bean,参考代码如下:
@Test public void testPerson() { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); Person p = ctx.getBean(Person.class); System.out.println(p); }
但是由于属于com.bean.Person的bean在IOC容器中不唯一,所以这里会抛出NoUniqueBeanDefinitionException异常。
由此我们可以总结getBean(String name)和getBean(Class type)的异同点。
3. getBean(String name,Class type)
这种方式比较适合当类型不唯一时,再通过id或者name来获取bean。
例如applicationContext.xml配置有如下bean:
<bean id="p1" class="com.bean.Person"> <property name="name" value="张三"/> <property name="age" value="18"/> </bean> <bean name="p2" class="com.bean.Person"> <property name="name" value="李四"/> <property name="age" value="20"/> </bean>
参考代码如下:
@Test public void testPerson() { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); Person p = ctx.getBean("p2",Person.class); System.out.println(p); }
4、getBean(String name,Object[] args)
这种方式本质还是通过bean的id或者name来获取bean,通过第二个参数Object[] args可以给bean的属性赋值,赋值的方式有两种:构造方法和工厂方法。但是通过这种方式获取的bean必须把scope属性设置为prototype,也就是非单例模式。
先在com.factory包下设计有如下的工厂类:
public class PersonFactory { //静态工厂注入 public static Person getPersonInstance(String name,int age)throws Exception { Person p = (Person)Class.forName("com.bean.Person").newInstance(); Method m = p.getClass().getMethod("setName", java.lang.String.class); m.invoke(p, name); m = p.getClass().getMethod("setAge", int.class); m.invoke(p, age); return p; } }
在applicationContext.xml中配置有如下bean:
<bean name="p3" class="com.bean.Person" scope="prototype"/>
获取bean的参考代码:
@Test public void testPerson() { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); Person p = (Person) ctx.getBean("p3",new Object[]{"王五",35}); System.out.println(p); }
3.2 延迟加载
@Autowired @Lazy FeignUserService feignUserService;
默认情况下,Spring会在应用程序上下文的启动时创建所有单例bean。这背后的原因很简单:立即避免和检测所有可能的错误,而不是在运行时。
但是,有些情况下我们需要创建一个bean,而不是在应用程序上下文启动时,而是在我们请求时。