之前的文章已经分析过了ribbon的流程,如果我们只使用ribbon来进行服务调用的话,就需要依赖RestTemplate,看起来可能是这样的
@RequestMapping(value = "/hello/{name}", method = RequestMethod.GET)
public String greeting(@PathVariable("name") String name) {
RestTemplate restTemplate = getRestTemplate();
return restTemplate.getForObject("http://ServiceName/sayHello/" + name, String.class);
}
这就跟我们平时使用Spring MVC的方式有点不太一样,感觉是不是有些别扭,你说我就想自动注入一个Service,然后在接口调用的时候直接使用 服务名.方法名() 的方式来调用,借助feign,就可以实现声明式的服务调用
具体实现起来也很简单,我相信大家应该也不需要我从feign的使用来一点点讲了,简单来说就是我们的启动类上需要加上@EnableFeignClients注解,然后添加一个接口,接口需要使用@FeignClient注解,然后直接调用这个接口里面的方法就可以实现我们所希望的声明式服务调用了
接下来就是具体的流程分析了,会贴出一部分的源码,如果不习惯看大段的源码可以直接跳到最后看总结
首先我们的入口是@EnableFeignClients注解,这个注解标注在启动类上,顾名思义就是启用@FeignClient
@Import(FeignClientsRegistrar.class)
public @interface EnableFeignClients {
}
不重要的代码我都已经省略了,重点就是上面@Import中的FeignClientsRegistrar类,里面有一个方法叫做registerBeanDefinitions
@Override
public void registerBeanDefinitions(AnnotationMetadata metadata,
BeanDefinitionRegistry registry) {
registerDefaultConfiguration(metadata, registry);
registerFeignClients(metadata, registry);
}
上面那个registerDefaultConfiguration方法看方法名猜测可能是注册默认的配置信息,不是我们所关注的,所以直接跳过,接下来是registerFeignClients这个方法,代码有点多,我这边就不贴出来了,主要是扫描@EnableFeignClients注解中的属性(一般这里面也不怎么配置)和@FeignClient注解(这里面有很多属性可以配置,一般来说name属性是会配置的,也就是标记所要调用接口的服务名)
registerFeignClients方法走完之后会调用registerFeignClient方法,在这个方法里面有一个关键步骤就是创建动态代理
private void registerFeignClient(BeanDefinitionRegistry registry,
AnnotationMetadata annotationMetadata, Map<String, Object> attributes) {
String className = annotationMetadata.getClassName();
BeanDefinitionBuilder definition = BeanDefinitionBuilder
.genericBeanDefinition(FeignClientFactoryBean.class);
validate(attributes);
definition.addPropertyValue("url", getUrl(attributes));
definition.addPropertyValue("path", getPath(attributes));
String name = getName(attributes);
definition.addPropertyValue("name", name);
definition.