溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Feign中EnableFeignClients的作用是什么

發(fā)布時(shí)間:2021-08-03 11:34:42 來源:億速云 閱讀:778 作者:Leah 欄目:大數(shù)據(jù)

這期內(nèi)容當(dāng)中小編將會(huì)給大家?guī)碛嘘P(guān)Feign中EnableFeignClients的作用是什么,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

    springcloud-openfeign-core-2.1.1.release.

    在springcloud中使用feign的時(shí)候,如下List-1,F(xiàn)eignClient的name是服務(wù)名稱,會(huì)自動(dòng)從Eureka中獲取物理地址。

    List-1

@FeignClient(name="UserProvider")
public interface UserProvider {
...
}

    Springcloud中這個(gè)是如何實(shí)現(xiàn)的?

    這個(gè)要從@EnableFeignClients入手,如下List-2,Import注解引入了FeignClientsRegistrar——實(shí)現(xiàn)了ImportBeanDefinitionRegistrar接口,這樣springboot會(huì)處理這個(gè)FeignClientsRegistrar。

    List-2

@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 {};
}

    如下圖1所示,F(xiàn)eignClientsRegistrar并沒有復(fù)雜的繼承關(guān)系,重點(diǎn)類看ImportBeanDefinitionRegistrar的registerBeanDefinitions實(shí)現(xiàn)。

      Feign中EnableFeignClients的作用是什么

                                                                                          圖1

    如下List-3,分倆個(gè)步驟,首先是registerDefaultConfiguration方法,將EnableFeignClients的defaultConfiguration注冊(cè)到Spring容器中;之后是registerFeignClients方法將FeignClient注解的接口注冊(cè)到Spring容器中。

    List-3

@Override
public void registerBeanDefinitions(AnnotationMetadata metadata,
        BeanDefinitionRegistry registry) {
    registerDefaultConfiguration(metadata, registry);
    registerFeignClients(metadata, registry);
}

    來看registerDefaultConfiguration方法,如下List-4,

  1. 獲取EnableFeignClients的所有屬性,之后如果含有defaultConfiguration,則將defaultConfiguration注冊(cè)到Spring容器中

  2. 方法registerClientConfiguration中,用Builder模式,構(gòu)造FeignClientSpecification類型的BeanDefinition。FeignClientSpecification實(shí)現(xiàn)了NamedContextFactory.Specification接口,屬性有個(gè)name和Class<?>類型的configuration。

    List-4

private void registerDefaultConfiguration(AnnotationMetadata metadata,
BeanDefinitionRegistry registry) {
    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"));
    }
}

private void registerClientConfiguration(BeanDefinitionRegistry registry, Object name,
Object configuration) {
    BeanDefinitionBuilder builder = BeanDefinitionBuilder
        .genericBeanDefinition(FeignClientSpecification.class);
    builder.addConstructorArgValue(name);
    builder.addConstructorArgValue(configuration);
    registry.registerBeanDefinition(
        name + "." + FeignClientSpecification.class.getSimpleName(),
        builder.getBeanDefinition());
}

    registerFeignClients方法中,實(shí)現(xiàn)則較為復(fù)雜,如下List-5

    List-5

public void registerFeignClients(AnnotationMetadata metadata,
    BeanDefinitionRegistry registry) {
    ClassPathScanningCandidateComponentProvider scanner = getScanner();//1
    scanner.setResourceLoader(this.resourceLoader);

    Set<String> basePackages;

    Map<String, Object> attrs = metadata
            .getAnnotationAttributes(EnableFeignClients.class.getName());//2
    AnnotationTypeFilter annotationTypeFilter = new AnnotationTypeFilter(
            FeignClient.class);//3
    final Class<?>[] clients = attrs == null ? null
            : (Class<?>[]) attrs.get("clients");
    if (clients == null || clients.length == 0) {//4
        scanner.addIncludeFilter(annotationTypeFilter);
        basePackages = getBasePackages(metadata);
    }
    else {//5
        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"));//6

                registerFeignClient(registry, annotationMetadata, attributes);//7
            }
        }
    }
}
  1. 1處獲取ClassPathScanner,用于掃描類路徑

  2. 2處獲取EnableFeignClients的所有屬性

  3. 3處構(gòu)造一個(gè)AnnotationTypeFilter,構(gòu)造方法參數(shù)是FeignClient,這個(gè)用于過濾出只含有FeignClient的類

  4. 獲得EnableFeignClients的clients屬性值,4處如果是空,則獲得EnableFeignClients所在的package路徑(如果沒有設(shè)置basePackageClasses)

  5. 5處,即EnableFeignClients的clients屬性不是空,則遍歷,放入集合中,同時(shí)獲取client所在的package路面,加入到basePacakges中;構(gòu)造AbstractClassTestingTypeFilter,這是增加一個(gè)過濾條件,即標(biāo)FeignClient注解的接口,必須在EnableFeignClients的clients中

  6. 遍歷basePackages,獲取每個(gè)package下的符合條件的類,得到對(duì)應(yīng)的beanDefinition,6處得到FeignClient的configuration值,通過FeignClientSpecification其注冊(cè)到spring容器中,有意思的是這里檢查了FeignClient注解的類須是接口,不然會(huì)報(bào)錯(cuò)。

  7. 7處將FeignClient注解的接口封裝到FeignClientFactoryBean中,F(xiàn)actoryBean大家懂的,Spring中接口都封裝到這個(gè)里面。

上述就是小編為大家分享的Feign中EnableFeignClients的作用是什么了,如果剛好有類似的疑惑,不妨參照上述分析進(jìn)行理解。如果想知道更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI