背景:
项目要导入feign的jar包,然后将jar包中的标有@feignClient注解的接口注入到spring容器中,别人在调用的时候可以直接从beanFactory.getBean(类全路径)去拿到feign接口对应的代理对象,然后反射进行远程调用。
难点1:
这边jar包可以重新上传,所以会有类重名的情况,使用appClassloader肯定不行的,所以每次加载都要新建一个classloader。这就导致spring框架在getBean(“类全路径名称”)去加载类的时候,找不到通过自定义类加载器加载进来的类。
解决办法:通过跟进源码发现,spring默认使用的类加载器实在项目启动的时候当前线程的类加载器,后面getBean反射加载类的时候也是使用的这个classLoader。
在spring著名的refresh方法中的第三步prepareBeanFactory(beanFactory);
中设置了spring的类加载器。也支持后期set。
// 获取spring上下文中的bean工厂
AnnotationConfigServletWebServerApplicationContext applicationContext = (AnnotationConfigServletWebServerApplicationContext) feignClientRegistrar.getApplicationContext();
DefaultListableBeanFactory beanFactory = applicationContext.getDefaultListableBeanFactory();
// 这个才是重点,不是beanFactory的setClassloader方法,设置的是不一样的。
applicationContext.setClassLoader(myClassloader);
难点2:
扫描feign接口注入到spring中,注入的肯定是feign接口的代理对象,因为要识别出接口上面的servicename,url之类的。
这个我是通过跟进@EnableFeignClients
里面引入的FeignClientsRegistrar类,去找到里面实现注册的部分,因为类的访问权限和源码实现逻辑不能扩展,所以我自己定义了一个类就是把源码拿过来改吧改吧,又加了一个方法里面实现图片中的for循环。