Feign源碼解析——執(zhí)行過程

Feign源碼解析——執(zhí)行過程

在上一篇Feign源碼解析——初始化流程中我們從注解為起點介紹了Feign是如何進行初始化參數(shù)、將Bean注入到Spring容器中的栖雾。接下來我們就介紹Feign是如何開始執(zhí)行的楞抡。

回顧

在上一篇中我們了解到了Feign將自身的參數(shù)注入到Spring容器中是分兩種類型進行注入的。

  • FeignClientSpecification:主要為名字和Configuration的對應(yīng)析藕,包括@FeignClient中的Configuration召廷,名字為value值,還包括@EnableFeignClients注解中的name和DefaultConfiguration對應(yīng)
  • FeignClientFactoryBean:它其中包含了所有@FeignClient注解上的參數(shù)账胧,他實現(xiàn)了Spring中的FactoryBean接口竞慢。他是一個工廠類,用于創(chuàng)建實例治泥。

不了解FactoryBean接口的筹煮,可以看如何使用Spring的FactoryBean接口

FeignClientFactoryBean介紹

作為一個實現(xiàn)了FactoryBean的工廠類,那么每次在Spring Context 創(chuàng)建實體類的時候會調(diào)用它的getObject()方法居夹。

class FeignClientFactoryBean implements FactoryBean<Object>, InitializingBean,
        ApplicationContextAware {
    private Class<?> type;

    private String name;

    private String url;

    private String path;

    private boolean decode404;

    private ApplicationContext applicationContext;

    private Class<?> fallback = void.class;

    private Class<?> fallbackFactory = void.class;
    -------其余代碼省略
    @Override
    public Object getObject() throws Exception {
        FeignContext context = applicationContext.getBean(FeignContext.class);
        Feign.Builder builder = feign(context);

        if (!StringUtils.hasText(this.url)) {
            String url;
            if (!this.name.startsWith("http")) {
                url = "http://" + this.name;
            }
            else {
                url = this.name;
            }
            url += cleanPath();
            return loadBalance(builder, context, new HardCodedTarget<>(this.type,
                    this.name, url));
        }
        if (StringUtils.hasText(this.url) && !this.url.startsWith("http")) {
            this.url = "http://" + this.url;
        }
        String url = this.url + cleanPath();
        Client client = getOptional(context, Client.class);
        if (client != null) {
            if (client instanceof LoadBalancerFeignClient) {
                // not lod balancing because we have a url,
                // but ribbon is on the classpath, so unwrap
                client = ((LoadBalancerFeignClient)client).getDelegate();
            }
            builder.client(client);
        }
        Targeter targeter = get(context, Targeter.class);
        return targeter.target(this, builder, context, new HardCodedTarget<>(
                this.type, this.name, url));
    }
    -------其余代碼省略
}

這里的getObject()其實就是將@FeinClient中設(shè)置value值進行組裝起來败潦,此時或許會有疑問,因為在配置FeignClientFactoryBean類時特意說過并沒有將Configuration傳過來吮播,那么Configuration中的屬性是如何配置的呢变屁?看其第一句是

FeignContext context = applicationContext.getBean(FeignContext.class);

意思就是從Spring容器中獲取FeignContext.class的類,我們可以看下這個類是從哪加載的意狠。我們可以看FeignAutoConfiguration此類粟关,源碼如下,我們可以看到在此類中自動配置了FeignContext類,并且將FeignClientSpecification類型的類全部加入此類的屬性中。

@Configuration
@ConditionalOnClass(Feign.class)
@EnableConfigurationProperties({FeignClientProperties.class, FeignHttpClientProperties.class})
public class FeignAutoConfiguration {

    @Autowired(required = false)
    private List<FeignClientSpecification> configurations = new ArrayList<>();

    @Bean
    public HasFeatures feignFeature() {
        return HasFeatures.namedFeature("Feign", Feign.class);
    }

    @Bean
    public FeignContext feignContext() {
        FeignContext context = new FeignContext();
        context.setConfigurations(this.configurations);
        return context;
    }

    @Configuration
    @ConditionalOnClass(name = "feign.hystrix.HystrixFeign")
    protected static class HystrixFeignTargeterConfiguration {
        @Bean
        @ConditionalOnMissingBean
        public Targeter feignTargeter() {
            return new HystrixTargeter();
        }
    }

    @Configuration
    @ConditionalOnMissingClass("feign.hystrix.HystrixFeign")
    protected static class DefaultFeignTargeterConfiguration {
        @Bean
        @ConditionalOnMissingBean
        public Targeter feignTargeter() {
            return new DefaultTargeter();
        }
    }

-------省略

}

至此我們已經(jīng)完成了配置屬性的裝配工作闷板,那么是如何執(zhí)行的呢澎灸?我們可以看getObject()最后一句可以看到返回了Targeter.target的方法。

return targeter.target(this, builder, context, new HardCodedTarget<>(
                this.type, this.name, url));

那么這個Targeter是哪來的遮晚?我們還是看上面的FeignAutoConfiguration類性昭,可以看到其中有兩個Targeter類,一個是DefaultTargeter县遣,一個是HystrixTargeter糜颠。當(dāng)配置了feign.hystrix.enabled= true的時候,Spring容器中就會配置HystrixTargeter此類萧求,如果為false那么Spring容器中配置的就是DefaultTargeter

我們以DefaultTargeter為例介紹一下接下啦是如何通過創(chuàng)建代理對象的

class DefaultTargeter implements Targeter {

    @Override
    public <T> T target(FeignClientFactoryBean factory, Feign.Builder feign, FeignContext context,
                        Target.HardCodedTarget<T> target) {
        return feign.target(target);
    }
}

public static class Builder {

    public <T> T target(Target<T> target) {
      return build().newInstance(target);
    }

    public Feign build() {
      SynchronousMethodHandler.Factory synchronousMethodHandlerFactory =
          new SynchronousMethodHandler.Factory(client, retryer, requestInterceptors, logger,
                                               logLevel, decode404);
      ParseHandlersByName handlersByName =
          new ParseHandlersByName(contract, options, encoder, decoder,
                                  errorDecoder, synchronousMethodHandlerFactory);
      return new ReflectiveFeign(handlersByName, invocationHandlerFactory);
    }
  }

查看ReflectiveFeign類中newInstance方法是返回一個代理對象

  public <T> T newInstance(Target<T> target) {
    Map<String, MethodHandler> nameToHandler = targetToHandlersByName.apply(target);
    Map<Method, MethodHandler> methodToHandler = new LinkedHashMap<Method, MethodHandler>();
    List<DefaultMethodHandler> defaultMethodHandlers = new LinkedList<DefaultMethodHandler>();

    for (Method method : target.type().getMethods()) {
      if (method.getDeclaringClass() == Object.class) {
        continue;
      } else if(Util.isDefault(method)) {
        DefaultMethodHandler handler = new DefaultMethodHandler(method);
        defaultMethodHandlers.add(handler);
        methodToHandler.put(method, handler);
      } else {
        methodToHandler.put(method, nameToHandler.get(Feign.configKey(target.type(), method)));
      }
    }
    //設(shè)置攔截器
    InvocationHandler handler = factory.create(target, methodToHandler);
    T proxy = (T) Proxy.newProxyInstance(target.type().getClassLoader(), new Class<?>[]{target.type()}, handler);

    for(DefaultMethodHandler defaultMethodHandler : defaultMethodHandlers) {
      defaultMethodHandler.bindTo(proxy);
    }
    return proxy;
  }

最終都是執(zhí)行了SynchronousMethodHandler攔截器中的invoke方法

@Override
  public Object invoke(Object[] argv) throws Throwable {
    RequestTemplate template = buildTemplateFromArgs.create(argv);
    Retryer retryer = this.retryer.clone();
    while (true) {
      try {
        return executeAndDecode(template);
      } catch (RetryableException e) {
        retryer.continueOrPropagate(e);
        if (logLevel != Logger.Level.NONE) {
          logger.logRetry(metadata.configKey(), logLevel);
        }
        continue;
      }
    }
  }

invoke方法方法主要是應(yīng)用 encoder其兴,decoder 以及 retry 等配置, 并且自身對于調(diào)用結(jié)果有一定的處理邏輯夸政。我們最關(guān)心的請求實現(xiàn)元旬,實際上是在組裝 SynchronousMethodHandler 的 client 參數(shù)上,即前面提到的守问,如果當(dāng)前路徑里面有 Ribbon匀归,就是 LoadBalancerFeignClient,如果沒有耗帕,根據(jù)配置生成 ApacheHttpClient 或者 OKHttpClient穆端。在 Ribbon 里面,實現(xiàn)了 Eureka 服務(wù)發(fā)現(xiàn)以及進行請求等動作兴垦。當(dāng)然 Ribbon 里面還帶了負(fù)載均衡邏輯徙赢。

SynchronousMethodHandler其實是不關(guān)心調(diào)用過程的,他只是處理調(diào)用的結(jié)果探越。調(diào)用過程是實現(xiàn)了Client的實現(xiàn)類來做的狡赐。例如RibbonLoadBalancerFeignClient

總結(jié)

到此為止钦幔,F(xiàn)eign的配置和執(zhí)行流程已經(jīng)簡單的說完了王暗。

調(diào)用接口為什么會直接發(fā)送請求婴程?

原因就是Spring掃描了@FeignClient注解巷疼,并且根據(jù)配置的信息生成代理類汽绢,調(diào)用的接口實際上調(diào)用的是生成的代理類。

Feign的整體工作流程

  1. 掃描@EnableFeignClients注解中配置包路徑卷玉。
  2. 掃描@FeignClient注解哨颂,并將注解配置的信息注入到Spring容器中,類型為FeignClientFactoryBean相种。
  3. 根據(jù)FeignClientFactoryBeangetObject()方法得到不同動態(tài)代理的類威恼。
  4. 根據(jù)不同的代理執(zhí)行不同的invoke()方法。

參考文章

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市箫措,隨后出現(xiàn)的幾起案子腹备,更是在濱河造成了極大的恐慌,老刑警劉巖斤蔓,帶你破解...
    沈念sama閱讀 218,122評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件植酥,死亡現(xiàn)場離奇詭異,居然都是意外死亡弦牡,警方通過查閱死者的電腦和手機友驮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來驾锰,“玉大人喊儡,你說我怎么就攤上這事〉揪荩” “怎么了?”我有些...
    開封第一講書人閱讀 164,491評論 0 354
  • 文/不壞的土叔 我叫張陵买喧,是天一觀的道長捻悯。 經(jīng)常有香客問我,道長淤毛,這世上最難降的妖魔是什么今缚? 我笑而不...
    開封第一講書人閱讀 58,636評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮低淡,結(jié)果婚禮上姓言,老公的妹妹穿的比我還像新娘。我一直安慰自己蔗蹋,他們只是感情好何荚,可當(dāng)我...
    茶點故事閱讀 67,676評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著猪杭,像睡著了一般餐塘。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上皂吮,一...
    開封第一講書人閱讀 51,541評論 1 305
  • 那天戒傻,我揣著相機與錄音,去河邊找鬼蜂筹。 笑死需纳,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的艺挪。 我是一名探鬼主播不翩,決...
    沈念sama閱讀 40,292評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了慌盯?” 一聲冷哼從身側(cè)響起周霉,我...
    開封第一講書人閱讀 39,211評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎亚皂,沒想到半個月后俱箱,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,655評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡灭必,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,846評論 3 336
  • 正文 我和宋清朗相戀三年狞谱,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片禁漓。...
    茶點故事閱讀 39,965評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡跟衅,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出播歼,到底是詐尸還是另有隱情伶跷,我是刑警寧澤,帶...
    沈念sama閱讀 35,684評論 5 347
  • 正文 年R本政府宣布秘狞,位于F島的核電站叭莫,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏烁试。R本人自食惡果不足惜雇初,卻給世界環(huán)境...
    茶點故事閱讀 41,295評論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望减响。 院中可真熱鬧靖诗,春花似錦、人聲如沸支示。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽悼院。三九已至伤为,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間据途,已是汗流浹背绞愚。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留颖医,地道東北人位衩。 一個月前我還...
    沈念sama閱讀 48,126評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像熔萧,于是被迫代替她去往敵國和親糖驴。 傳聞我的和親對象是個殘疾皇子僚祷,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,914評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 這段時間沒事的時候總是習(xí)慣性的拿起手機,刷刷抖音贮缕,在上面我發(fā)現(xiàn)現(xiàn)在很多的正能量再出現(xiàn)辙谜,包括很多的媒體,新聞感昼,段視頻...
    書等一個人閱讀 163評論 0 0
  • 一装哆,齊全了 需要衛(wèi)生許可證,工商營業(yè)執(zhí)照定嗓,稅務(wù)登記證蜕琴,稅務(wù)登記證包括營業(yè)執(zhí)照,房屋租賃合同宵溅,房租產(chǎn)權(quán)證凌简,法人...
    478339124d9c閱讀 79評論 0 0
  • 感恩感謝公開大場的精彩!
    清凈心學(xué)閱讀 74評論 0 0
  • 人確實是奇怪的東西恃逻,不想多想雏搂,不然必亂。 期望太久的寇损,比如特別想念一個人畔派。。润绵。。思念太久胞谈,就會有所期望尘盼,期望和實際...
    teresadai閱讀 213評論 0 0
  • GEO數(shù)據(jù)庫格式2018-01-27_164221.jpg 參考的文章AKAP95 regulates splic...
    韋巍_e3f3閱讀 681評論 0 0