Spring Cloud (三、服务调用-Ribbon、OpenFeign)

spring cloud

1、Ribbon

Spring Cloud Ribbon 是基于 Netflix Ribbon 实现的一套 客户端负载均衡 的工具。
简单的说,Ribbon是Netflix发布的开源项目,主要功能是提供客户端的软件负载均衡算法和服务调用。Ribbon客户端组件提供一系列完善的配置项如连接超时,重试等。简单的说,就是在配置文件中列出Load Balancer(简称LB)后面所有的机器,Ribbon会自动的帮助你基于某种规则(如简单轮询,随机连接等)去连接这些机器。我们很容易使用Ribbon实现自定义的负载均衡算法。

1.1、概述

  1. 官网资料

    官网地址:https://github.com/Netflix/ribbon/wiki/Getting-Started
    Ribbon目前也进入维护模式
    在这里插入图片描述

  2. 未来替换方案:Spring cloud Loadbalance
    在这里插入图片描述3. 功能
    LB负载均衡(Load Balance)是什么
    简单的说就是将用户的请求平摊的分配到多个服务上,从而达到系统的HA(高可用)。
    常见的负载均衡有软件Nginx,LVS,硬件 F5等。

    Ribbon本地负载均衡客户端 VS Nginx服务端负载均衡区别
    Nginx是服务器负载均衡,客户端所有请求都会交给nginx,然后由nginx实现转发请求。即负载均衡是由服务端实现的。

    Ribbon本地负载均衡,在调用微服务接口时候,会在注册中心上获取注册信息服务列表之后缓存到JVM本地,从而在本地实现RPC远程服务调用技术。

    集中式LB
    即在服务的消费方和提供方之间使用独立的LB设施(可以是硬件,如F5, 也可以是软件,如nginx), 由该设施负责把访问请求通过某种策略转发至服务的提供方;

    进程内LB
    将LB逻辑集成到消费方,消费方从服务注册中心获知有哪些地址可用,然后自己再从这些地址中选择出一个合适的服务器。
    Ribbon就属于进程内LB,它只是一个类库,集成于消费方进程,消费方通过它来获取到服务提供方的地址。

1.2、架构说明及负载均衡演示

Ribbon其实就是一个软负载均衡的客户端组件,
他可以和其他所需请求的客户端结合使用,和eureka结合只是其中的一个实例。
Ribbon在工作时分成两步
第一步先选择 EurekaServer ,它优先选择在同一个区域内负载较少的server.
第二步再根据用户指定的策略,在从server取到的服务注册列表中选择一个地址。
其中Ribbon提供了多种策略:比如轮询、随机和根据响应时间加权。
在这里插入图片描述
之前的例子 8001、8002 能实现负载均衡,这是因为 spring-cloud-starter-netflix-eureka-client 自带了 spring-cloud-starter-ribbon 引用

1.2.1、RestTemplate的使用

  1. 官网地址:
    https://docs.spring.io/spring-framework/docs/5.2.2.RELEASE/javadoc-api/org/springframework/web/client/RestTemplate.html

  2. getForObject方法/getForEntity方法
    返回对象为响应体中数据转化成的对象,基本上可以理解为Json
    在这里插入图片描述
    返回对象为ResponseEntity对象,包含了响应中的一些重要信息,比如响应头、响应状态码、响应体等
    在这里插入图片描述
    在这里插入图片描述

  3. GET请求方法
    T getForObject(String url, Class responseType, Object… uriVariables);

    T getForObject(String url, Class responseType, Map<String, ?> uriVariables);

    T getForObject(URI url, Class responseType);

    ResponseEntity getForEntity(String url, Class responseType, Object… uriVariables);

    ResponseEntity getForEntity(String url, Class responseType, Map<String, ?> uriVariables);

    ResponseEntity getForEntity(URI var1, Class responseType);

  4. POST请求方法
    T postForObject(String url, @Nullable Object request, Class responseType, Object… uriVariables);

    T postForObject(String url, @Nullable Object request, Class responseType, Map<String, ?> uriVariables);

    T postForObject(URI url, @Nullable Object request, Class responseType);

    ResponseEntity postForEntity(String url, @Nullable Object request, Class responseType, Object… uriVariables);

    ResponseEntity postForEntity(String url, @Nullable Object request, Class responseType, Map<String, ?> uriVariables);

    ResponseEntity postForEntity(URI url, @Nullable Object request, Class responseType);

1.3、Ribbon核心组件IRule

IRule:根据特定算法中从服务列表中选取一个要访问的服务
在这里插入图片描述

  • com.netflix.loadbalancer.RoundRobinRule
    轮询
  • com.netflix.loadbalancer.RandomRule
    随机
  • com.netflix.loadbalancer.RetryRule
    先按照RoundRobinRule的策略获取服务,如果获取服务失败则在指定时间内会进行重试,获取可用的服务
  • WeightedResponseTimeRule
    对RoundRobinRule的扩展,响应速度越快的实例选择权重越大,越容易被选择
  • BestAvailableRule
    会先过滤掉由于多次访问故障而处于断路器跳闸状态的服务,然后选择一个并发量最小的服务
  • AvailabilityFilteringRule
    先过滤掉故障实例,再选择并发较小的实例
  • ZoneAvoidanceRule
    默认规则,复合判断server所在区域的性能和server的可用性选择服务器

1.2.1、如何替换负载均衡模式

  1. 使用 cloud-consumer-order80 进行修改

    官方文档明确给出了警告:
    这个自定义配置类不能放在@ComponentScan所扫描的当前包下以及子包下,
    否则我们自定义的这个配置类就会被所有的Ribbon客户端所共享,达不到特殊化定制的目的了。
    在这里插入图片描述

  2. 新建package,com.atguigu.myrule

  3. MySelfRule

    package com.atguigu.myrule;
    
    import com.netflix.loadbalancer.IRule;
    import com.netflix.loadbalancer.RandomRule;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    
    /**
     * @auther zzyy
     * @create 2020-02-02 14:31
     */
    @Configuration
    public class MySelfRule
    {
        @Bean
        public IRule myRule()
        {
            return new RandomRule();//定义为随机
        }
    }
    
  4. 启动类添加注释

    package com.atguigu.springcloud;
    
    import com.atguigu.myrule.MySelfRule;
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
    import org.springframework.cloud.netflix.ribbon.RibbonClient;
    
    /**
     * @auther zzyy
     * @create 2020-01-28 16:18
     * 在启动该微服务的时候就能去加载我们的自定义Ribbon配置类,从而使配置生效,形如:
     */
    @SpringBootApplication
    @EnableEurekaClient
    @RibbonClient(name = "CLOUD-PAYMENT-SERVICE",configuration=MySelfRule.class)
    public class OrderMain80
    {
        public static void main(String[] args)
        {
            SpringApplication.run(OrderMain80.class,args);
        }
    }
    

    注意:@RibbonClient(name = "CLOUD-PAYMENT-SERVICE",configuration=MySelfRule.class) name的值一定要大写,小写的话则不会生效

  5. 测试
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

1.4、仿照轮询模式,写一个负载均衡功能

  1. 轮训模式的核心算法为:访问的次数 % 服务器集群数量 = 实际的服务器列表的下标

  2. 核心代码

    RoundRobinRule.java

    //
    // Source code recreated from a .class file by IntelliJ IDEA
    // (powered by FernFlower decompiler)
    //
    
    package com.netflix.loadbalancer;
    
    import com.netflix.client.config.IClientConfig;
    import java.util.List;
    import java.util.concurrent.atomic.AtomicInteger;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    
    public class RoundRobinRule extends AbstractLoadBalancerRule {
        private AtomicInteger nextServerCyclicCounter;
        private static final boolean AVAILABLE_ONLY_SERVERS = true;
        private static final boolean ALL_SERVERS = false;
        private static Logger log = LoggerFactory.getLogger(RoundRobinRule.class);
    
        public RoundRobinRule() {
            this.nextServerCyclicCounter = new AtomicInteger(0);
        }
    
        public RoundRobinRule(ILoadBalancer lb) {
            this();
            this.setLoadBalancer(lb);
        }
    
        public Server choose(ILoadBalancer lb, Object key) {
            if (lb == null) {
                log.warn("no load balancer");
                return null;
            } else {
                Server server = null;
                int count = 0;
    
                while(true) {
                    if (server == null && count++ < 10) {
                        List<Server> reachableServers = lb.getReachableServers();
                        List<Server> allServers = lb.getAllServers();
                        int upCount = reachableServers.size();
                        int serverCount = allServers.size();
                        if (upCount != 0 && serverCount != 0) {
                            int nextServerIndex = this.incrementAndGetModulo(serverCount);
                            server = (Server)allServers.get(nextServerIndex);
                            if (server == null) {
                                Thread.yield();
                            } else {
                                if (server.isAlive() && server.isReadyToServe()) {
                                    return server;
                                }
    
                                server = null;
                            }
                            continue;
                        }
    
                        log.warn("No up servers available from load balancer: " + lb);
                        return null;
                    }
    
                    if (count >= 10) {
                        log.warn("No available alive servers after 10 tries from load balancer: " + lb);
                    }
    
                    return server;
                }
            }
        }
    
        private int incrementAndGetModulo(int modulo) {
            int current;
            int next;
            do {
                current = this.nextServerCyclicCounter.get();
                next = (current + 1) % modulo;
            } while(!this.nextServerCyclicCounter.compareAndSet(current, next));
    
            return next;
        }
    
        public Server choose(Object key) {
            return this.choose(this.getLoadBalancer(), key);
        }
    
        public void initWithNiwsConfig(IClientConfig clientConfig) {
        }
    }
    
    

    涉及到自旋锁、JUC等知识。。。

1.4.1、仿照轮询模式,写一个负载均衡功能,步骤及核心代码编写

    1. 启动 7001、7002 Eureka集群
    1. 8001、8002 服务端的controller进行修改,增加返回端口号功能。

    PaymentController.java

       @GetMapping(value = "/payment/lb")
       public String getPaymentLB() {
           return serverPort;
       }
    
    1. 80服务端改造
      在这里插入图片描述
      • ApplicationContextBean去掉注解@LoadBalanced
        package com.atguigu.springcloud.config;
        
        import org.springframework.cloud.client.loadbalancer.LoadBalanced;
        import org.springframework.context.annotation.Bean;
        import org.springframework.context.annotation.Configuration;
        import org.springframework.web.client.RestTemplate;
        
        @Configuration
        public class ApplicationContextConfig {
        
            @Bean
            //@LoadBalanced //使用@LoadBalanced注解赋予RestTemplate负载均衡的能力
            public RestTemplate getRestTemplate() {
        
                return new RestTemplate();
        
            }
        
        }
        
      • LoadBalancer接口
        package com.atguigu.springcloud.lb;
        
        import org.springframework.cloud.client.ServiceInstance;
        
        import java.util.List;
        
        public interface LoadBalancer {
            ServiceInstance serviceInstance(List<ServiceInstance> serviceInstances);
        }
        
      • MyLB
        package com.atguigu.springcloud.lb;
        
        import lombok.extern.slf4j.Slf4j;
        import org.springframework.cloud.client.ServiceInstance;
        import org.springframework.stereotype.Component;
        
        import java.util.List;
        import java.util.concurrent.atomic.AtomicInteger;
        
        @Slf4j
        @Component
        public class MyLB implements LoadBalancer {
        
            private AtomicInteger atomicInteger = new AtomicInteger(0);
        
            public final int getAndIncrement() {
        
                int current;
                int next;
                do {
                    current = this.atomicInteger.get();
                    next = current >= Integer.MAX_VALUE ? 0 : current + 1;
        
                } while (!this.atomicInteger.compareAndSet(current, next));
        
                log.info("**********next:" + next);
                return next;
        
            }
        
            @Override
            public ServiceInstance serviceInstance(List<ServiceInstance> serviceInstances) {
                return serviceInstances.get(getAndIncrement() % serviceInstances.size());
            }
        }
        
        
      • OrderController
        package com.atguigu.springcloud.controller;
        
        
        import com.atguigu.springcloud.lb.LoadBalancer;
        import entities.CommResult;
        import entities.Payment;
        import lombok.extern.slf4j.Slf4j;
        import org.springframework.beans.factory.annotation.Autowired;
        import org.springframework.cloud.client.ServiceInstance;
        import org.springframework.cloud.client.discovery.DiscoveryClient;
        import org.springframework.web.bind.annotation.GetMapping;
        import org.springframework.web.bind.annotation.PathVariable;
        import org.springframework.web.bind.annotation.RestController;
        import org.springframework.web.client.RestTemplate;
        
        import javax.annotation.Resource;
        import java.net.URI;
        import java.util.List;
        
        @RestController
        @Slf4j
        public class OrderController {
        
            public static final String PAYMENT_URL = "http://CLOUD-PAYMENT-SERVICE";
            @Autowired
            private RestTemplate restTemplate;
        
            @Autowired
            private DiscoveryClient discoveryClient;
        
            @Autowired
            private LoadBalancer loadBalancer;
        
            @GetMapping("/consumer/payment/create")
            public CommResult<Payment> create(Payment payment) {
        
                log.info("*********->OrderController.create");
                return restTemplate.postForObject(PAYMENT_URL + "/payment/create", payment, CommResult.class);
        
            }
        
            @GetMapping("/consumer/payment/get/{id}")
            public CommResult<Payment> getPayment(@PathVariable("id") Long id) {
        
                log.info("*********->OrderController.getPayment");
                return restTemplate.getForObject(PAYMENT_URL + "/payment/get/" + id, CommResult.class);
        
            }
        
            @GetMapping("/consumer/payment/lb")
            public String getPaymenLB() {
                List<ServiceInstance> serviceInstances = discoveryClient.getInstances("CLOUD-PAYMENT-SERVICE");
                if (serviceInstances != null && serviceInstances.size() > 0) {
                    ServiceInstance serviceInstance = loadBalancer.serviceInstance(serviceInstances);
                    URI uri = serviceInstance.getUri();
                    return restTemplate.getForObject(uri + "/payment/lb", String.class);
                } else {
                    return null;
                }
            }
        }
        
  1. 测试
    在这里插入图片描述

2、OpenFeign

官网解释:
https://cloud.spring.io/spring-cloud-static/Hoxton.SR1/reference/htmlsingle/#spring-cloud-openfeign

Feign是一个声明式WebService客户端。使用Feign能让编写Web Service客户端更加简单。
它的使用方法是定义一个服务接口然后在上面添加注解。Feign也支持可拔插式的编码器和解码器。Spring Cloud对Feign进行了封装,使其支持了Spring MVC标准注解和HttpMessageConverters。Feign可以与Eureka和Ribbon组合使用以支持负载均衡

Feign是一个声明式的Web服务客户端,让编写Web服务客户端变得非常容易,只需创建一个接口并在接口上添加注解即可

GitHub源码:
https://github.com/spring-cloud/spring-cloud-openfeign

Feign能干什么
Feign旨在使编写Java Http客户端变得更容易。
前面在使用Ribbon+RestTemplate时,利用RestTemplate对http请求的封装处理,形成了一套模版化的调用方法。但是在实际开发中,由于对服务依赖的调用可能不止一处,往往一个接口会被多处调用,所以通常都会针对每个微服务自行封装一些客户端类来包装这些依赖服务的调用。所以,Feign在此基础上做了进一步封装,由他来帮助我们定义和实现依赖服务接口的定义。在Feign的实现下,我们只需创建一个接口并使用注解的方式来配置它(以前是Dao接口上面标注Mapper注解,现在是一个微服务接口上面标注一个Feign注解即可),即可完成对服务提供方的接口绑定,简化了使用Spring cloud Ribbon时,自动封装服务调用客户端的开发量。

Feign集成了Ribbon
利用Ribbon维护了Payment的服务列表信息,并且通过轮询实现了客户端的负载均衡。而与Ribbon不同的是,通过feign只需要定义服务绑定接口且以声明式的方法,优雅而简单的实现了服务调用

Feign和OpenFeign两者区别

FeignOpenFeign
Feign是Spring Cloud组件中的一个轻量级RESTful的HTTP服务客户端
Feign内置了Ribbon,用来做客户端负载均衡,去调用服务注册中心的服务。Feign的使用方式是:使用Feign的注解定义接口,调用这个接口,就可以调用服务注册中心的服务
OpenFeign是Spring Cloud 在Feign的基础上支持了SpringMVC的注解,如@RequesMapping等等。OpenFeign的@FeignClient可以解析SpringMVC的@RequestMapping注解下的接口,并通过动态代理的方式产生实现类,实现类中做负载均衡并调用其他服务。

2.1、OpenFeign使用步骤

添加依赖组件:

<!--openfeign-->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>

使用注释:
@FeignClient、@EnableFeignClients

  • 新建cloud-consumer-fegin-order80
    在这里插入图片描述

  • 修改POM

    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
        <modelVersion>4.0.0</modelVersion>
        <parent>
            <groupId>com.atguigu.springcloud</groupId>
            <artifactId>cloud2020</artifactId>
            <version>1.0-SNAPSHOT</version>
        </parent>
    
        <artifactId>cloud-consumer-fegin-order80</artifactId>
    
        <dependencies>
            <!--openfeign-->
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-starter-openfeign</artifactId>
            </dependency>
            <!--eureka client-->
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
            </dependency>
            <!-- 引入自己定义的api通用包,可以使用Payment支付Entity -->
            <dependency>
                <groupId>com.atguigu.springcloud</groupId>
                <artifactId>cloud-api-commons</artifactId>
                <version>${project.version}</version>
            </dependency>
            <!--web-->
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-web</artifactId>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-actuator</artifactId>
            </dependency>
            <!--一般基础通用配置-->
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-devtools</artifactId>
                <scope>runtime</scope>
                <optional>true</optional>
            </dependency>
            <dependency>
                <groupId>org.projectlombok</groupId>
                <artifactId>lombok</artifactId>
                <optional>true</optional>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-test</artifactId>
                <scope>test</scope>
            </dependency>
        </dependencies>
    
    </project>
    
  • 写YML

    spring:
      application:
        name: cloud-consumer-fegin-order80
    server:
      port: 80
    eureka:
      client:
        #表示是否将自己注册进EurekaServer默认为true。
        register-with-eureka: true
        #是否从EurekaServer抓取已有的注册信息,默认为true。单节点无所谓,集群必须设置为true才能配合ribbon使用负载均衡
        fetchRegistry: true
        service-url:
          #defaultZone: http://localhost:7001/eureka
          defaultZone: http://eureka7001.com:7001/eureka,http://eureka7002.com:7002/eureka  # 集群版
    	```
    
  • 主启动

    package com.atguigu.springcloud;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.cloud.openfeign.EnableFeignClients;
    
    @SpringBootApplication
    @EnableFeignClients
    public class OrderFeginMain80 {
    
        public static void main(String[] args) {
    
            SpringApplication.run(OrderFeginMain80.class, args);
    
        }
    
    }
    
  • 业务类

    package com.atguigu.springcloud;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.cloud.openfeign.EnableFeignClients;
    
    @SpringBootApplication
    @EnableFeignClients
    public class OrderFeginMain80 {
    
        public static void main(String[] args) {
    
            SpringApplication.run(OrderFeginMain80.class, args);
    
        }
    
    }
    
    package com.atguigu.springcloud.controller;
    
    import com.atguigu.springcloud.service.PaymentFeginService;
    import entities.CommResult;
    import entities.Payment;
    import lombok.extern.slf4j.Slf4j;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RestController;
    
    @RestController
    @Slf4j
    public class OrderFeginController {
    
        @Autowired
        private PaymentFeginService paymentFeginService;
    
        @GetMapping("/consumer/payment/get/{id}")
        public CommResult<Payment> getPaymentById(@PathVariable("id") Long id){
            log.info("请求开始--->");
            return paymentFeginService.getPaymentById(id);
        }
    }
    
  • 测试
    在这里插入图片描述
    在这里插入图片描述



    在这里插入图片描述

2.2、OpenFeign设置超时时间

默认Feign客户端只等待一秒钟,但是服务端处理需要超过1秒钟,导致Feign客户端不想等待了,直接返回报错。
为了避免这样的情况,有时候我们需要设置Feign客户端的超时控制。
yml文件中开启配置

spring:
 application:
   name: cloud-consumer-fegin-order80
server:
 port: 80
eureka:
 client:
   #表示是否将自己注册进EurekaServer默认为true。
   register-with-eureka: true
   #是否从EurekaServer抓取已有的注册信息,默认为true。单节点无所谓,集群必须设置为true才能配合ribbon使用负载均衡
   fetchRegistry: true
   service-url:
     #defaultZone: http://localhost:7001/eureka
     defaultZone: http://eureka7001.com:7001/eureka,http://eureka7002.com:7002/eureka  # 集群版
#设置feign客户端超时时间(OpenFeign默认支持ribbon)
ribbon:
#指的是建立连接所用的时间,适用于网络状况正常的情况下,两端连接所用的时间
 ReadTimeout: 5000
#指的是建立连接后从服务器读取到可用资源所用的时间
 ConnectTimeout: 5000

在测试的时候发现,使用集群时,如果8001超时了之后,再发起其他请求,也不会分发到8001,当经过一段时间后,初步感觉应该是心跳时间间隔,再发起请求又可以分发到8001。

2.3、OpenFeign日志打印功能

Feign 提供了日志打印功能,我们可以通过配置来调整日志级别,从而了解 Feign 中 Http 请求的细节。
说白了就是对Feign接口的调用情况进行监控和输出。

日志的级别:

  • NONE:默认的,不显示任何日志;
  • BASIC:仅记录请求方法、URL、响应状态码及执行时间;
  • HEADERS:除了 BASIC 中定义的信息之外,还有请求和响应的头信息;
  • FULL:除了 HEADERS 中定义的信息之外,还有请求和响应的正文及元数据。

如何启用日志功能:

  • 新建configuration组件
    package com.atguigu.springcloud.cfgbeans;
    
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import feign.Logger;
    
    /**
     * @auther zzyy
     * @create 2019-11-10 17:00
     */
    @Configuration
    public class FeignConfig
    {
        @Bean
        Logger.Level feignLoggerLevel()
        {
            return Logger.Level.FULL;
        }
    }
    
  • 配置YML文件
    logging: level:com.atguigu.springcloud.service 这个为要启用日志的目录
    spring:
      application:
        name: cloud-consumer-fegin-order80
    server:
      port: 80
    eureka:
      client:
        #表示是否将自己注册进EurekaServer默认为true。
        register-with-eureka: true
        #是否从EurekaServer抓取已有的注册信息,默认为true。单节点无所谓,集群必须设置为true才能配合ribbon使用负载均衡
        fetchRegistry: true
        service-url:
          #defaultZone: http://localhost:7001/eureka
          defaultZone: http://eureka7001.com:7001/eureka,http://eureka7002.com:7002/eureka  # 集群版
    
    ribbon:
      #指的是建立连接所用的时间,适用于网络状况正常的情况下,两端连接所用的时间
      ReadTimeout: 5000
      #指的是建立连接后从服务器读取到可用资源所用的时间
      ConnectTimeout: 5000
    logging:
      level:
        com.atguigu.springcloud.service: debug
    
  • 查看日志:
    在这里插入图片描述
### 回答1: ork.cloud:spring-cloud-starter-openfeign:jar是一个基于Spring Cloud的开源库,用于简化微服务架构中服务之间的调用和通信。 OpenFeign是一个声明式的Web服务客户端,它简化了编写HTTP请求的代码,使得服务调用更加简单和高效。它是Spring Cloud提供的一个集成了Ribbon和Hystrix的库,可以方便地与其他微服务组件集成使用。 使用OpenFeign,我们可以通过编写接口的方式来定义服务间的调用,而不需要关注底层的HTTP请求。通过注解配置,OpenFeign会自动根据接口定义生成对应的代理类,并且提供了负载均衡、断路器等功能,方便处理高并发和服务故障的情况。 在微服务架构中,服务之间的调用是非常频繁的,而且随着微服务的增多,手动编写HTTP请求的代码会变得非常繁琐和容易出错。使用OpenFeign可以大大简化服务之间的调用流程,提高开发效率和代码质量。 总结来说,ork.cloud:spring-cloud-starter-openfeign:jar是一个方便而强大的库,可以帮助我们简化微服务架构中服务之间的调用和通信,并且提供了负载均衡和断路器等功能,能够提高系统的可靠性和性能。 ### 回答2: ork.cloud:spring-cloud-starter-openfeign:jar是一个基于Spring Cloud的开源项目,它提供了一种方便的编写和调用RESTful服务的方式。Feign是一个声明式的Web服务客户端,它可以简化HTTP请求的处理和封装,使得开发者可以更加专注于业务逻辑的编写。 使用Spring Cloud Starter OpenFeign可以快速地编写和调用其他微服务。它通过注解的方式将HTTP请求映射到对应的方法上,自动进行了服务的发现和负载均衡。 Feign支持多种请求方式,包括GET、POST、PUT、DELETE等,还可以使用@PathVariable、@RequestParam等注解处理路径参数和查询参数。Feign还支持对请求体进行处理,可以将请求体转换成Java对象,方便业务逻辑的处理。 在使用Feign时,不需要手动编写HTTP请求的代码,只需要定义一个接口并使用Feign的注解进行标记即可。Feign会根据注解生成代理对象来完成请求的发送和接收。这样可以大大简化开发的工作量,并且使得代码更加清晰易读。 Spring Cloud Starter OpenFeign还集成了Ribbon和Hystrix,这使得我们在使用Feign时可以实现负载均衡和熔断的功能。即使请求的目标服务发生宕机或故障,也能够保证系统的高可用性和稳定性。 总之,Spring Cloud Starter OpenFeign是一个非常实用和方便的工具,可以简化微服务架构下的服务调用,并提供了负载均衡和熔断等功能。它的使用可以加快开发速度,提高系统的可靠性和稳定性。 ### 回答3: spring-cloud-starter-openfeign是一个开源的Spring Cloud组件,用于简化在微服务架构中进行远程服务调用的过程。它基于Netflix的Feign库进行开发,提供了一种声明式的、基于接口的远程服务调用方式,可以方便地实现服务之间的通信和数据交互。 ork.cloud:spring-cloud-starter-openfeign:jar是spring-cloud-starter-openfeign组件的一个特定版本的jar包。在使用Spring Boot构建的项目中,可以通过引入这个jar包来集成并使用spring-cloud-starter-openfeign组件,从而简化远程服务调用的代码编写和配置。 使用spring-cloud-starter-openfeign,我们只需要定义一个接口,通过注解的方式声明远程服务的地址和调用方法,然后在需要调用远程服务的地方直接调用这个接口的方法即可。Spring Cloud会根据注解信息自动进行服务发现和负载均衡,将我们的调用请求转发到对应的服务实例上。 该jar包中除了包含spring-cloud-starter-openfeign的核心功能外,还可能包含一些额外的依赖库或工具,以及特定版本的相关代码和配置文件。通过引入这个jar包,我们可以一键集成和启用spring-cloud-starter-openfeign组件,省去了手动添加依赖和配置的步骤,能够更快速地搭建起微服务架构中的服务调用机制。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值