美文网首页
Ribbon自定义负载均衡算法

Ribbon自定义负载均衡算法

作者: HelloDake | 来源:发表于2018-11-09 11:34 被阅读0次

    Ribbon内置7种负载均衡算法,通过IRule接口的choose()方法来实现不同的负载均衡算法,我们通过自己实现choose()方法的方式来达到自定义负载均衡策略的目的

    IRule接口

    需求:每台机器访问5次后随即轮询

    1:消费者主启动类

    添加注解@RibbonClient(name="MICROSERVICECLOUD-DEPT",configuration=MySelfRule.class),用来指定我们自定义的Ribbon配置类

    2:创建配置类与choose()方法实现类

    @Configuration
    public class MySelfRule
    {
        @Bean
        public IRule myRule()
        {
            return new RandomRule_ZY();// 我自定义为每台机器5次
        }
    }
    
    

    注: 自定义策略源码是在RandomRule算法上做了修改

    public class RandomRule_ZY extends AbstractLoadBalancerRule
    {
    
        // total = 0 // 当total==5以后,我们指针才能往下走,
        // index = 0 // 当前对外提供服务的服务器地址,
        // total需要重新置为零,但是已经达到过一个5次,我们的index = 1
        // 分析:我们5次,但是微服务只有8001 8002 8003 三台,OK?
        //
    
    
        private int total = 0;          // 总共被调用的次数,目前要求每台被调用5次
        private int currentIndex = 0;   // 当前提供服务的机器号
    
        public Server choose(ILoadBalancer lb, Object key)
        {
            if (lb == null) {
                return null;
            }
            Server server = null;
    
            while (server == null) {
                if (Thread.interrupted()) {
                    return null;
                }
    //获取可用的Server集合
                List<Server> upList = lb.getReachableServers();
    //获取全部server集合
                List<Server> allList = lb.getAllServers();
    
                int serverCount = allList.size();
                if (serverCount == 0) {
                    /*
                     * No servers. End regardless of pass, because subsequent passes only get more
                     * restrictive.
                     */
                    return null;
                }
    
    //          int index = rand.nextInt(serverCount);// java.util.Random().nextInt(3);
    //          server = upList.get(index);
    
    //服务调用计数
    //          private int total = 0;          // 总共被调用的次数,目前要求每台被调用5次
    //          private int currentIndex = 0;   // 当前提供服务的机器号
                if(total < 5)
                {
                    server = upList.get(currentIndex);
                    total++;
                }else {
                    total = 0;
                    currentIndex++;
                    if(currentIndex >= upList.size())
                    {
                      currentIndex = 0;
                    }
                }
    
    
                if (server == null) {
                    /*
                     * The only time this should happen is if the server list were somehow trimmed.
                     * This is a transient condition. Retry after yielding.
                     */
                    Thread.yield();
                    continue;
                }
    
                if (server.isAlive()) {
                    return (server);
                }
    
                // Shouldn't actually happen.. but must be transient or a bug.
                server = null;
                Thread.yield();
            }
    
            return server;
    
        }
    
        @Override
        public Server choose(Object key)
        {
            return choose(getLoadBalancer(), key);
        }
    
        @Override
        public void initWithNiwsConfig(IClientConfig clientConfig)
        {
            // TODO Auto-generated method stub
    
        }
    
    }
    

    注意:自定义配置类不能放在@ComponentScan所扫描的当前包下以及子包下,否则我们自定义的这个配置类就会被所有的Ribbon客户端所共享,也就是说,我们达不到特殊化定制的目的了。

    image.png

    相关文章

      网友评论

          本文标题:Ribbon自定义负载均衡算法

          本文链接:https://www.haomeiwen.com/subject/gcbnxqtx.html