欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解openfeign集成spring?cloud?loadbalancer實現(xiàn)負載均衡流程

 更新時間:2023年07月06日 09:45:07   作者:子瞻  
這篇文章主要介紹了openfeign集成spring?cloud?loadbalancer實現(xiàn)負載均衡流程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

Feignclient負載均衡中如何起作用

當我們在項目中調用自己實現(xiàn)的Feignclient負載均衡中如何起作用?請看下圖:

圖一

在圖一中我們可以發(fā)現(xiàn),在ConsumerController中調用自定義的DemoFeignClient方法時,通過spring容器中對DemoFeignclient的代理類的調用最終通過feign.SynchronousMethodHandler.invoke()->openfeign.loadbalancer.execute()->org.springframework.cloud.loadbalancer.blocking.client.FeignBlockingLoadBalancerClient.choose()->org.springframework.cloud.loadbalancer.core.RoundRobinLoadBalancer.choose()這樣的調用鏈,最終執(zhí)行spring cloud loadbalaner中的輪詢負載均橫策略!

然而,上圖中的這些關鍵類是如何組合起來發(fā)揮作用呢?請我們一起繼續(xù)分析!spring cloud項目啟動后,spring容器解析并加載LoadBalancerClientConfiguration.java配置文件(如下圖所示)

圖二

然后將"reactorServiceInstanceLoadBalancer"注冊到beanDefinitionMap中。
然后會掃描我們聲明的controller,因為conroller中注入了DemoFeignClient,因此spring容器會遞歸創(chuàng)建DemoFeignClient,創(chuàng)建DemoFiegnClient過程中會通過AbstractAutowireCapableBeanFactory.obtainFromSupplier()注入instanceSupplier實例,也就是FeignClientFactoryBean實例!

主要邏輯

下面是org.springframework.cloud.openfeign.FeignClientsRegistrar.registerFeignClient()方法主要邏輯:

private void registerFeignClient(BeanDefinitionRegistry registry, AnnotationMetadata annotationMetadata,
            Map<String, Object> attributes) {
          //忽略非核心代碼片段
            BeanDefinitionBuilder definition = BeanDefinitionBuilder.genericBeanDefinition(clazz, () -> {
            //設置url
            factoryBean.setUrl(getUrl(beanFactory, attributes));
            //設置路徑
            factoryBean.setPath(getPath(beanFactory, attributes));
            //decode404 布爾值            factoryBean.setDecode404(Boolean.parseBoolean(String.valueOf(attributes.get("decode404"))));
            Object fallback = attributes.get("fallback");
            //設置fallback
            if (fallback != null) {
                factoryBean.setFallback(fallback instanceof Class ? (Class<?>) fallback
                        : ClassUtils.resolveClassName(fallback.toString(), null));
            }
            //設置fallback工廠類
            Object fallbackFactory = attributes.get("fallbackFactory");
            if (fallbackFactory != null) {
                factoryBean.setFallbackFactory(fallbackFactory instanceof Class ? (Class<?>) fallbackFactory
                        : ClassUtils.resolveClassName(fallbackFactory.toString(), null));
            }
            //實例化我們聲明的DemoClient
            return factoryBean.getObject();
        });
}

這還沒有完,org.springframework.cloud.openfeign.FeignClientFactoryBean.getObject()中最終執(zhí)行l(wèi)oadBalance();在loadBalance()中實例化FeignInvocationHandler;

調用DemoFeignClient中方法

在項目啟動成功之后,我們調用DemoFeignClient中的方法時

圖三

通過圖三可以發(fā)現(xiàn),描述DemoFeignClient的RootBeanDefinition的類中還有一個叫“instanceSupplier”的類型屬性,它的值是“FeignClientsRegistrar$lambda”,那么這個類時什么時候被注入進來的呢?答案就是上面分析項目啟動過程中spring容器根據(jù)RootBeanDefinition對DemoFeignClient的描述,通過對"FeignClientsRegistrar$lambda"的調用完成對DemoFeignClient實例的創(chuàng)建。

在完成創(chuàng)建DemoFeignClient實例后,我們調用org.springframework.cloud.openfeign.loadbalancer。

FeignBlockingLoadBalancerClient.execute()執(zhí)行負載均衡策略時,執(zhí)行到如下代碼塊

Set<LoadBalancerLifecycle> supportedLifecycleProcessors = LoadBalancerLifecycleValidator
                .getSupportedLifecycleProcessors(
                        loadBalancerClientFactory.getInstances(serviceId, LoadBalancerLifecycle.class),
                        RequestDataContext.class, ResponseData.class, ServiceInstance.class);

"loadBalancerClientFactory.getInstances(serviceId, LoadBalancerLifecycle.class)"邏輯會獲取LoadBalancer實例,最終調用LoadBalancerClientConfiguration.java中的reactorServiceInstanceLoadBalancer()實現(xiàn)創(chuàng)建負載均衡實例邏輯

@Bean
    @ConditionalOnMissingBean
    public ReactorLoadBalancer<ServiceInstance> reactorServiceInstanceLoadBalancer(Environment environment,
            LoadBalancerClientFactory loadBalancerClientFactory) {
        String name = environment.getProperty(LoadBalancerClientFactory.PROPERTY_NAME);
        //創(chuàng)建負載均衡實例,而且我們發(fā)現(xiàn)默認的負載均衡實例是輪詢負載均衡實例
        return new RoundRobinLoadBalancer(
                loadBalancerClientFactory.getLazyProvider(name, ServiceInstanceListSupplier.class), name);
    }

ok,截止到目前為止,我們的負載均衡實例已經創(chuàng)建完成!

這樣一來,我們通過openfeign進行遠程調用時,通過下圖的調用鏈

圖四

FeignInvocationHandler.invoke()->SynchronousMethodHandler.invoke()->org.springframework.cloud.openfeign.loadbalancer.FeignBlockingLoadBalancerClient.execute()->org.springframework.cloud.loadbalancer.blocking.client.BlockingLoadBalancerClient.choose()->org.springframework.cloud.loadbalancer.core.RoundRobinLoadBalancer.choose(),這樣的調用鏈最終根據(jù)RoundRobinLoadBalancer.choose()獲取服務實例。

public Mono<Response<ServiceInstance>> choose(Request request) {
        ServiceInstanceListSupplier supplier = serviceInstanceListSupplierProvider
                .getIfAvailable(NoopServiceInstanceListSupplier::new);
        //返回服務實例
        return supplier.get(request).next()
                .map(serviceInstances -> processInstanceResponse(supplier, serviceInstances));
    }

拿到服務實例后,F(xiàn)eignBlockingLoadBalancerClient.execute()中最終通過LoadBalancerUtils.executeWithLoadBalancerLifecycleProcessing()獲取響應結果!

下面是具體代碼邏輯:

static Response executeWithLoadBalancerLifecycleProcessing(Client feignClient, Request.Options options,
            Request feignRequest, org.springframework.cloud.client.loadbalancer.Request lbRequest,
            org.springframework.cloud.client.loadbalancer.Response<ServiceInstance> lbResponse,
            Set<LoadBalancerLifecycle> supportedLifecycleProcessors, boolean loadBalanced, boolean useRawStatusCodes)
            throws IOException {
        supportedLifecycleProcessors.forEach(lifecycle -> lifecycle.onStartRequest(lbRequest, lbResponse));
        try {
            //通過openfeign調用,獲取響應結果
            Response response = feignClient.execute(feignRequest, options);
            if (loadBalanced) {
                supportedLifecycleProcessors.forEach(
                        lifecycle -> lifecycle.onComplete(new CompletionContext<>(CompletionContext.Status.SUCCESS,
                                lbRequest, lbResponse, buildResponseData(response, useRawStatusCodes))));
            }
            return response;
        }
        catch (Exception exception) {
            if (loadBalanced) {
                supportedLifecycleProcessors.forEach(lifecycle -> lifecycle.onComplete(
                        new CompletionContext<>(CompletionContext.Status.FAILED, exception, lbRequest, lbResponse)));
            }
            throw exception;
        }
    }

以上就是詳解openfeign集成spring cloud loadbalancer實現(xiàn)負載均衡流程的詳細內容,更多關于spring cloud loadbalancer的資料請關注腳本之家其它相關文章!

相關文章

  • cookie、session和java過濾器結合實現(xiàn)登陸程序

    cookie、session和java過濾器結合實現(xiàn)登陸程序

    這篇文章主要為大家詳細介紹了cookie、session和java過濾器結合實現(xiàn)登陸程序的具體代碼,感興趣的朋友可以參考一下
    2016-05-05
  • Java接口返回省市區(qū)樹形結構的實現(xiàn)

    Java接口返回省市區(qū)樹形結構的實現(xiàn)

    本文主要介紹了Java接口返回省市區(qū)樹形結構的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-01-01
  • 解析Java中PriorityQueue優(yōu)先級隊列結構的源碼及用法

    解析Java中PriorityQueue優(yōu)先級隊列結構的源碼及用法

    優(yōu)先級隊列是一種隊列結構,是0個或多個元素的集合,每個元素都有一個優(yōu)先權,PriorityQueue被內置于JDK中,本文就來解析Java中PriorityQueue優(yōu)先級隊列結構的源碼及用法.
    2016-05-05
  • SpringBoot選擇自有bean優(yōu)先加載實現(xiàn)方法

    SpringBoot選擇自有bean優(yōu)先加載實現(xiàn)方法

    在一些需求中,可能存在某些場景,比如先加載自己的bean,然后自己的bean做一些DB操作,初始化配置問題,然后后面的bean基于這個配置文件,繼續(xù)做其他的業(yè)務邏輯。因此有了本文的這個題目
    2023-03-03
  • Mybatis優(yōu)化檢索的方法詳解

    Mybatis優(yōu)化檢索的方法詳解

    MyBatis是一款優(yōu)秀的基于Java的持久層框架,它可以將 SQL 語句和數(shù)據(jù)庫中的記錄映射成為 Java 對象,并且支持靈活的 SQL 查詢語句,在Mybatis中,可以使用動態(tài)SQL來靈活構造SQL語句,從而滿足各種不同的檢索需求,本文介紹Mybatis如何優(yōu)化檢索,需要的朋友可以參考下
    2024-05-05
  • Java+Selenium實現(xiàn)文件上傳下載功能詳解

    Java+Selenium實現(xiàn)文件上傳下載功能詳解

    這篇文章主要介紹了java代碼如何利用selenium操作瀏覽器上傳和下載文件功能,文中的示例代碼講解詳細,具有一定的借鑒價值,需要的可以參考一下
    2023-01-01
  • 基于Luhn算法的銀行卡校驗規(guī)則

    基于Luhn算法的銀行卡校驗規(guī)則

    這篇文章主要為大家介紹了基于Luhn算法的銀行卡校驗規(guī)則,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05
  • Spring中的集合注入代碼實例

    Spring中的集合注入代碼實例

    這篇文章主要介紹了Spring中的集合注入代碼實例,集合注入是指在Spring框架中,通過配置文件或注解的方式將集合類型的數(shù)據(jù)注入到Bean中,集合類型包括List、Set、Map和Properties等,需要的朋友可以參考下
    2023-11-11
  • SpringMVC實現(xiàn)登錄與注冊功能的詳細步驟

    SpringMVC實現(xiàn)登錄與注冊功能的詳細步驟

    本文介紹了如何通過Maven配置依賴,創(chuàng)建前端登錄和注冊頁面,并實現(xiàn)后端邏輯,詳細步驟包括配置文件、創(chuàng)建User類、配置中文過濾器及DispatcherServlet,并使用Spring?MVC和JQuery處理前端請求,需要的朋友可以參考下
    2024-11-11
  • Java在Excel中添加水印的實現(xiàn)(單一水印、平鋪水印)

    Java在Excel中添加水印的實現(xiàn)(單一水印、平鋪水印)

    這篇文章主要介紹了Java在Excel中添加水印的實現(xiàn)(單一水印、平鋪水印),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-04-04

最新評論