spring 的 ApplicationContext.getBean(type)无法获取bean,报错

少年乘勇气,百战过乌孙。这篇文章主要讲述spring 的 ApplicationContext.getBean(type)无法获取bean,报错相关的知识,希望能为你提供帮助。
具体问题请看    https://q.cnblogs.com/q/108101/
研究了两天: 经过上文中的排除法: 造成问题的原因是要获取的bean 中 有被切入的方法。。
就是可能该类会使用反射生成一个类。。
怎么测试呢?
想到 @Autowired  和 @Resource  这两个注解。。
他们会通过 类型 和 名称去找容器中对应 的 bean 。。
于是在controller 中使用 这个注解 注入 zaService;  
报错了  : 

Caused by: org.springframework.beans.factory.BeanNotOfRequiredTypeException: Bean named \'zaSerivce\' is expected to be of type \'com.mytest.context.impl.ZaSerivce\' but was actually of type \'com.sun.proxy.$Proxy15\' at org.springframework.beans.factory.support.DefaultListableBeanFactory.checkBeanNotOfRequiredType(DefaultListableBeanFactory.java:1510) at org.springframework.beans.factory.support.DefaultListableBeanFactory.checkBeanNotOfRequiredType(DefaultListableBeanFactory.java:1517) at org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoMatchingBeanFound(DefaultListableBeanFactory.java:1489) at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:1104) at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:1066) at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:585) ... 67 more

这就说明了两个问题: 1. 为什么 byTpe 无法获取bean ,,因为它的类型根本就不是    com.mytest.context.impl.ZaSerivce 而是      com.sun.proxy.$Proxy15
2.  byName 虽然可以获取但是会发现它的类型也已经不是 ZaService 了。。 
ApplicationContext context =ApplicationContextUtils.context; ZaService z = (ZaService) context.getBean("zaService"); java.lang.ClassCastException: com.sun.proxy.$Proxy15 cannot be cast to com.mytest.context.impl.ZaService


 
 
【spring 的 ApplicationContext.getBean(type)无法获取bean,报错】解决办法: 1. 通过byType 获取bean 时参数传 接口类的class ..    不足: 只能有一个实现类。
2.  byName 可以正常传参数 bean 的首字母小写。。 但是只能转换为接口类 
IZaService z = (IZaService) context.getBean("zaService");
 
而我的业务参数是  实现类的全路径名,所以特别适合 class.forName();   然后 byType .. 可惜用不了。。 只能退而求其次了把参数 调整为 接口的全路径名。。但是只能有一个实现类。。
至于为什么被 切 的 类 在spring 容器中的 type 变了, 那可能要考虑代理反射,个人观点是 spring 默认使用jdk 动态代理,这种方式会生成一个全新的类,比如本例的
$Proxy15, 然后通过这个代理类方法前后编入代码,然后调用原始类的方法。。 说远了,。。。。

注意 : spring 还有一种代理方式: CGLIB 是不会生成新的类,那如果不生成新的类,就不会出现上述的问题了,,现在验证下(spring 对没有实现接口的类的代理方式是 CGLib )

我们只要把 IZaServiec 去掉就行了不要Zaserivce 实现它,试试,,日,,真的正常了。。

@Service public class ZaService{@TestAnn /*@Override*/ public void aa() {} }@ResponseBody @RequestMapping("/context") public String context(){ ApplicationContext context =ApplicationContextUtils.context; ZaService z = (ZaService) context.getBean(ZaService.class); System.out.println(z); z.aa(); return "context"; }com.mytest.context.impl.ZaService@7cb0dfee < > > > > > > > > > > > > > > > > > > > > > > > after< > < > < > < > < > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > >





 

    推荐阅读