莫要嘲笑看原始碼的朋友,現在面試只是八股文是靠不住了,更多是問專案問題以及原始碼及問題。我也是逼不得已,不然誰想造輪子,很累人又枯燥!
個人覺得看原始碼的前提是得會用,用熟了可以猜猜別人是怎麼實現的,如果有相關官方文件那就在看看官方文件。
不過,可惜的是很多官方文件寫得很爛,讓你看了會那種雲裡霧裡的。
最近我在研究openfeign
原始碼的時候,發現在原始碼中有個關鍵註解:@Import
。
專案啟動類別:
/** * @author tianwc 公众号:java后端技术全栈、面试专栏 * @version 1.0.0 * @date 2023年07月07日 16:47 * 在线刷题 1200+题和1000+篇干货文章:<a href="http://woaijava.cc/">博客地址</a> */ @EnableFeignClients(basePackages = {"com.tian.feign"}) @SpringBootApplication() public class MqApplication { public static void main(String[] args) { SpringApplication.run(MqApplication.class, args); } }
然後,就是我們的feignclient
介面:
/** * @author tianwc 公众号:java后端技术全栈、面试专栏 * @version 1.0.0 * @date 2023年07月07日 16:47 * 在线刷题 1200+题和1000+篇干货文章:<a href="http://woaijava.cc/">博客地址</a> */ @FeignClient(contextId = "userFeignClient", value = "charge-user-service") public interface UserFeignClient { /** * 邀请成功增加收益 * * @param invitedDto 邀请增加收益 * @return 邀请成功 */ @PostMapping("/user/invited/register") CommonResult<Boolean> invitedRegister(@RequestBody InvitedDto invitedDto); }
使用案例:
/** * @author tianwc 公众号:java后端技术全栈、面试专栏 * @version 1.0.0 * @date 2023年07月07日 16:47 * 在线刷题 1200+题和1000+篇干货文章:<a href="http://woaijava.cc/">博客地址</a> */ @RestController @RequestMapping("/user") public class UserController { @Resource UserFeignClient userFeignClient; @PostMapping("/invited") public CommonResult invitedRegister(){ //省略不想搞的代码 return userFeignClient.invitedRegister(invitedDto); } }
從上面在的程式碼中,我們可以看出openfeign
關鍵程式碼有:
@EnableFeignClients(basePackages = {"com.tian.feign"})
@FeignClient(contextId = "userFeignClient", value = "charge-user-service")
################################################################################################################################。 ##
userFeignClient.invitedRegister(invitedDto);
EnableFeignClients
@EnableFeignClients
这个注解在启动类上,我们肯定要重点关注。
小技巧:凡是以
@Enable
开头的各种注解基本上都是开启xxxx
。比如:@EnableFeignClients
表示开启feign客户端。
我们进入@EnableFeignClients
中
/** * @author tianwc 公众号:java后端技术全栈、面试专栏 * @version 1.0.0 * @date 2023年07月07日 16:47 * 在线刷题 1200+题和1000+篇干货文章:<a href="http://woaijava.cc/">博客地址</a> */ @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.TYPE) @Documented @Import(FeignClientsRegistrar.class) public @interface EnableFeignClients { String[] value() default {}; String[] basePackages() default {}; Class<?>[] basePackageClasses() default {}; Class<?>[] defaultConfiguration() default {}; Class<?>[] clients() default {}; }
我们通常只需要关心属性basePackages
,表示我们需要扫描的包目录。
如果既没有指定
basePackages
,也没有指定basePackageClasses
,则采用启动类所在的目录作为包扫描路径。默认是这种情况。
本文重点来了,在这个注解@EnableFeignClients
上有个注解@Import(FeignClientsRegistrar.class)
,这里到底是有什么作用?
@Import()
#@Import()
註解是在spring 3.0版本中引入的,字面意義就是導入.
@Import
註解的全類名是org.springframework.context.annotation.Import
。其只有一個預設的value屬性,該屬性類型為Class<?>[]
,表示可以傳入一個或多個Class物件。
透過註解可以看出,該註解有如下作用:
可以導入一個或多個元件類別(通常是@Configuration配置類別)該註解的功能與Spring XML中的<import/>
元素相同。可以導入@Configuration
配置類別、ImportSelect
和ImportBeanDefinitionRegistrar
的實作類別。
從spring 4.2版本開始,也可以引用常規元件類別(普通類別),該功能類似於AnnotationConfigApplicationContext.register()
方法。
此註解可以在類別中聲明,也可以在元註解中聲明。如果需要匯入XML或其他非@Configuration
定義的資源,可以使用@ImportResource
註解。
通常有三種使用方式:
@Import
一个普通类 spring会将该类加载到spring容器中@Import
一个类,该类实现了ImportBeanDefinitionRegistrar
接口,在重写的registerBeanDefinitions
方法里面,能拿到BeanDefinitionRegistry
的注册器,能手工往beanDefinitionMap
中注册 beanDefinition
@Import
一个类 该类实现了ImportSelector
重写selectImports
方法该方法返回了String[]数组的对象,数组里面的类都会注入到spring容器当中。下面我们来聊聊@Import
在openfeign的这里是起到什么作用。
openfeign
中作用回答上面的代码里
@Import(FeignClientsRegistrar.class)
这里导入的是FeignClientsRegistrar
类,我们再来看看他的类关系图:
从类关系图来看,FeignClientsRegistrar
实现了ImportBeanDefinitionRegistrar
接口。再结合@Import
的三种使用方式中的第二种方式,能手工往beanDefinitionMap
中注册 beanDefinition
。
/** * @author tianwc 公众号:java后端技术全栈、面试专栏 * @version 1.0.0 * @date 2023年07月07日 16:47 * 在线刷题 1200+题和1000+篇干货文章:<a href="http://woaijava.cc/">博客地址</a> */ @Override public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) { registerDefaultConfiguration(metadata, registry); registerFeignClients(metadata, registry); }
这个方法registerBeanDefinitions()
是feign的核心入口方法,其中会做两件事:
注册默认的配置和注册所有的FeignClient。
registerDefaultConfiguration(metadata, registry)
这个方法是负责注册OpenFeign
的默认配置 ,逻辑相对简单:
private void registerDefaultConfiguration(AnnotationMetadata metadata, BeanDefinitionRegistry registry) { //获取@EnableFeignClients的全部属性 //@EnableFeignClients(basePackages = {"com.tian.feign"}) //这里的basePackages就是我们指定的熟悉 Map<String, Object> defaultAttrs = metadata .getAnnotationAttributes(EnableFeignClients.class.getName(), true); if (defaultAttrs != null && defaultAttrs.containsKey("defaultConfiguration")) { String name; if (metadata.hasEnclosingClass()) { name = "default." + metadata.getEnclosingClassName(); } else { name = "default." + metadata.getClassName(); } registerClientConfiguration(registry, name, defaultAttrs.get("defaultConfiguration")); } }
defaultAttrs
中内容如下:
但是这里我们只关注defaultConfiguration,我们并有对其进行设置,所以我们可以忽略他。重点是下面这个方法。
registerFeignClients(metadata, registry)
这里就是项目启动时和openfeign相关的核心代码,这也是@EnableFeignClients
和@FeignClient
两个注解关联起来的地方。
我们进入源码中:
public void registerFeignClients(AnnotationMetadata metadata, BeanDefinitionRegistry registry) { ClassPathScanningCandidateComponentProvider scanner = getScanner(); scanner.setResourceLoader(this.resourceLoader); Set<String> basePackages; Map<String, Object> attrs = metadata .getAnnotationAttributes(EnableFeignClients.class.getName()); AnnotationTypeFilter annotationTypeFilter = new AnnotationTypeFilter( FeignClient.class); final Class<?>[] clients = attrs == null ? null : (Class<?>[]) attrs.get("clients"); if (clients == null || clients.length == 0) { scanner.addIncludeFilter(annotationTypeFilter); basePackages = getBasePackages(metadata); } else { final Set<String> clientClasses = new HashSet<>(); basePackages = new HashSet<>(); for (Class<?> clazz : clients) { basePackages.add(ClassUtils.getPackageName(clazz)); clientClasses.add(clazz.getCanonicalName()); } AbstractClassTestingTypeFilter filter = new AbstractClassTestingTypeFilter() { @Override protected boolean match(ClassMetadata metadata) { String cleaned = metadata.getClassName().replaceAll("\\$", "."); return clientClasses.contains(cleaned); } }; scanner.addIncludeFilter( new AllTypeFilter(Arrays.asList(filter, annotationTypeFilter))); } for (String basePackage : basePackages) { Set<BeanDefinition> candidateComponents = scanner .findCandidateComponents(basePackage); for (BeanDefinition candidateComponent : candidateComponents) { if (candidateComponent instanceof AnnotatedBeanDefinition) { // verify annotated class is an interface AnnotatedBeanDefinition beanDefinition = (AnnotatedBeanDefinition) candidateComponent; AnnotationMetadata annotationMetadata = beanDefinition.getMetadata(); Assert.isTrue(annotationMetadata.isInterface(), "@FeignClient can only be specified on an interface"); Map<String, Object> attributes = annotationMetadata .getAnnotationAttributes( FeignClient.class.getCanonicalName()); String name = getClientName(attributes); registerClientConfiguration(registry, name, attributes.get("configuration")); registerFeignClient(registry, annotationMetadata, attributes); } } } }
代码一行一行看是不是觉得很累,我给你总结好了。
上面的方法分为以下七个步骤:
@EnableFeignClients
註解的所有屬性,主要為了拿到掃描套件路徑(basePackages
);@EnableFeignClients
註解中配置clients屬性,所以會進入到clients屬性為空時的邏輯;方法獲取掃描器:
ClassPathScanningCandidateComponentProvider,並將上下文
AnnotationConfigServletWebServerApplicationContext作為掃描器的
ResourceLoader;
新增一個註解過濾器(AnnotationTypeFilter),只過濾出包含
@FeignClient註解的
BeanDefinition;
##再透過
getBasePackages(metadata)scanner.findCandidateComponents(basePackage)
方法從套件路徑下掃描出所有標註了@FeignClient
註解並符合條件組裝的介面;FeignClientConfiguration
在BeanDefinitionRegistry
中註冊一下,再對FeignClient
做真正的註冊操作。 在openfeign
原始碼中的@Import註解在這裡的作用就是將掃描到帶有FeignClient
註解的全部介面類別以bean的形式註冊到spring IOC容器中。
再來強調@Import註解使用方式:
@Import
一個普通類別spring會將該類別載入到spring容器中@Import
一個類,該類別實作了ImportBeanDefinitionRegistrar
接口,在重寫的registerBeanDefinitions
方法裡面,能拿到BeanDefinitionRegistry
的註冊器,可以手工往 beanDefinitionMap
中註冊beanDefinition
#@Import
一個類別該類別實作了ImportSelector
重寫selectImports
方法此方法傳回了String[]陣列的對象,陣列裡面的類別都會注入到spring容器當中。 好了,今天就分享這麼多。這個openfeign裡面還有很多很有意思的地方,我們下次再分享吧!
以上是Spring Cloud原始碼分析:第一篇的詳細內容。更多資訊請關注PHP中文網其他相關文章!