2020-09-11服务调用-Ribbon

2020-09-12  本文已影响0人  喵喵不吃鱼哦

作用:负载均衡
负载均衡:
LoadBalance包含集中式和LB和进程内LB
其中集中式LB式独立与消费者与服务提供方的,消费者通过LB获取知道自己应该使用哪一个服务提供方。
进程内LB是集成于消费者一方中,LB注册进入Eureka Server,获知可用服务提供方,然后直接对服务提供方进行调用。

使用:
1.POM引入

        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-ribbon</artifactId>
        </dependency>

当使用Eureka时候自带了ribbon引用


Eureka中ribbon

2.IRule:根据特定算法从服务列表中选取一个要访问的服务
注意IRule的自定义配置类不能放在@ComponentScan所能扫描到的地方,不然配置将会被所有的Ribbon客户端所共享,无法达到特殊化定制的目的与效果。
自定义配置类

@Configuration//添加此注解,表明为配置类,使得spring自动扫描识别
public class MyRule {
    @Bean
    public IRule getIRule(){
        return new RandomRule();//从默认的轮询定义为随机
        //return new RandomRule();//一定时限内循环重试
        //return new RoundRobinRule();//轮询
    }
}

主启动类添加@RibbonClient注解,value表示注册的Eureka服务注册中心,configuration表示采取的自定义配置

@SpringBootApplication
@EnableEurekaClient
@RibbonClient(value = "CLOUD-ORDER-SERVICE",configuration = MyRule.class)
public class OrderMain80 {
    public static void main(String[] args){
        SpringApplication.run(OrderMain80.class,args);
    }
}

可以配合RestTemplate来进行测试与使用

@Configuration
public class ApplicationContextConfig {
    @Bean
//    开启集群的restTemplate负载均衡,轮询
    @LoadBalanced
    public RestTemplate getRestTemplate(){
        return new RestTemplate();
    }
}

调用:

…………省略…………
public static final String PAYMENT_URL = "http://CLOUD-ORDER-SERVICE";
…………省略…………
@GetMapping("/consumer/payment")
    public CommonResult<Payment> create(Payment payment){
        return restTemplate.postForObject(PAYMENT_URL+"/payment/",payment,CommonResult.class);
    }
…………省略…………

手写Ribbon负载均衡算法
1.在ApplicationContextConfig 配置类中去掉@LoadBalance注解
2.创建LoadBanner接口

public interface LoadBalancer {
    ServiceInstance instance(List<ServiceInstance> serviceInstances);
}

3.实现该接口

@Component
public class MyLB implements LoadBalancer {
    //利用java并发原子类AtomicInteger创建一个对象,初始值为 0
    private AtomicInteger atomicInteger = new AtomicInteger(0);

    public final int getAndIncrement(){
        int current;//上一次的数值 初始默认 0
        int next;//返回值 初始默认 0
        do {
            current = this.atomicInteger.get();
            next = current >= 5 ? 0: current +1;
            System.out.println("*****current = " + current);
            System.out.println("*****next = " + next);
        }while(!this.atomicInteger.compareAndSet(current,next));
//        this.atomicInteger.compareAndSet(current,next) 意思是判断current的数值是否等于内存中指定位置的值(this.),如果相等,
//        返回true,并且将next复制到内存中替换原来的值,如果不等,则返回false,并且不会进行赋值替换
        System.out.println("******next = " + next);
        return next;
    }//以上方法用途是生成1,2,3,4,5,1,2,3,4,5,1,2,3,4,5......无论怎样都不会出现重复(每个12345可能顺序不同,可能是13245,但不会出现重复,为啥?因为原子)
    @Override
    public ServiceInstance instance(List<ServiceInstance> serviceInstances) {
        int index = getAndIncrement() % serviceInstances.size();
        return serviceInstances.get(index);
    }
}

4.改造服务提供者Controller方法

@RestController
@Slf4j
public class PaymentController
{
    @Value("${server.port}")
    private String serverPort;

    @GetMapping(value = "/payment/lb")
    public String getPaymentLB(){
        return serverPort;
    }
}

5.消费者Controller

@RestController
@Slf4j
public class OrderController {
//单机
//    public static final String PAYMENT_URL = "http://localhost:8001";
//集群
    public static final String PAYMENT_URL = "http://CLOUD-ORDER-SERVICE";

    @Resource
    private LoadBalancer loadBalancer;

    @Resource
    private DiscoveryClient discoveryClient;

    @Resource
    private RestTemplate restTemplate;

    @GetMapping(value = "/consumer/payment/lb")
    public String getPaymentLB(){
        List<ServiceInstance> instances = discoveryClient.getInstances("CLOUD-ORDER-SERVICE");
        if (instances == null || instances.size() <= 0) {
            return null;
        }
        ServiceInstance serviceInstance = loadBalancer.instance(instances);
        URI uri = serviceInstance.getUri();
        return restTemplate.getForObject(uri + "/payment/lb", String.class);
    }
}
上一篇 下一篇

猜你喜欢

热点阅读