服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 使用Java实现5种负载均衡算法实例

使用Java实现5种负载均衡算法实例

2022-01-24 12:58微小猴 Java教程

负载均衡指由多台服务器以对称的方式组成一个服务器集合,每台服务器都具有等价的地位,都可以单独对外提供服务而无须其他服务器的辅助,这篇文章主要给大家介绍了关于使用Java实现5种负载均衡算法的相关资料,需要的朋友可以

前言

负载均衡是为了解决并发情况下,多个请求访问,把请求通过提前约定好的规则转发给各个server。其中有好几个种经典的算法。在用java代码编写这几种算法之前,先来了解一下负载均衡这个概念。

 

概念

负载均衡是将客户端请求访问,通过提前约定好的规则转发给各个server。其中有好几个种经典的算法,下面我们用Java实现这几种算法。

使用Java实现5种负载均衡算法实例

 

几种负载均衡算法图例

使用Java实现5种负载均衡算法实例

主要的负载均衡算法是图中这些,在代码实现之前,我们先简单回顾一下他们的概念。

 

轮询算法

轮询算法按顺序把每个新的连接请求分配给下一个服务器,最终把所有请求平分给所有的服务器。

优点:绝对公平

缺点:无法根据服务器性能去分配,无法合理利用服务器资源。

package com.monkeyjava.learn.basic.robin;

import com.google.common.collect.Lists;

import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class TestRound {

  private Integer  index = 0;
  private List<String> ips = Lists.newArrayList("192.168.1.1", "192.168.1.2", "192.168.1.3");


  public String roundRobin(){
      String serverIp;
      synchronized(index){
          if (index >= ips.size()){
              index = 0;
          }
          serverIp= ips.get(index);
          //轮询+1
          index ++;
      }
      return serverIp;
  }

  public static void main(String[] args) {
      TestRound testRoundRobin =new TestRound();
      for (int i=0;i< 10 ;i++){
          String serverIp= testRoundRobin.roundRobin();
          System.out.println(serverIp);
      }
  }
}

输出结果:

192.168.1.1
192.168.1.2
192.168.1.3
192.168.1.1
192.168.1.2
192.168.1.3
192.168.1.1
192.168.1.2
192.168.1.3
192.168.1.1

 

加权轮询法

该算法中,每个机器接受的连接数量是按权重比例分配的。这是对普通轮询算法的改进,比如你可以设定:第三台机器的处理能力是第一台机器的两倍,那么负载均衡器会把两倍的连接数量分配给第3台机器,轮询可以将请求顺序按照权重分配到后端。

package com.monkeyjava.learn.basic.robin;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class TestWeight {
  private Integer index = 0;
  static Map<String, Integer> ipMap=new HashMap<String, Integer>(16);
  static {
      // 1.map, key-ip,value-权重
      ipMap.put("192.168.1.1", 1);
      ipMap.put("192.168.1.2", 2);
      ipMap.put("192.168.1.3", 4);

  }

  public List<String> getServerIpByWeight() {
      List<String> ips = new ArrayList<String>(32);
      for (Map.Entry<String, Integer> entry : ipMap.entrySet()) {
          String ip = entry.getKey();
          Integer weight = entry.getValue();
          // 根据权重不同,放入list 中的数量等同于权重,轮询出的的次数等同于权重
          for (int ipCount =0; ipCount < weight; ipCount++) {
              ips.add(ip);
          }
      }
      return ips;
  }

  public String weightRobin(){
      List<String> ips = this.getServerIpByWeight();
      if (index >= ips.size()){
          index = 0;
      }
      String serverIp= ips.get(index);
      index ++;
      return  serverIp;
  }

  public static void main(String[] args) {
      TestWeight testWeightRobin=new TestWeight();
      for (int i =0;i< 10 ;i++){
          String server=testWeightRobin.weightRobin();
          System.out.println(server);
      }
  }
}

输出结果:

192.168.1.1
192.168.1.3
192.168.1.3
192.168.1.3
192.168.1.3
192.168.1.2
192.168.1.2
192.168.1.1
192.168.1.3
192.168.1.3

 

加权随机法

获取带有权重的随机数字,随机这种东西,不能看绝对,只能看相对,我们不用index 控制下标进行轮询,只用random 进行随机取ip,即实现算法。

package com.monkeyjava.learn.basic.robin;

import java.util.*;

public class TestRandomWeight {

  static Map<String, Integer> ipMap=new HashMap<String, Integer>(16);
  static {
      // 1.map, key-ip,value-权重
      ipMap.put("192.168.1.1", 1);
      ipMap.put("192.168.1.2", 2);
      ipMap.put("192.168.1.3", 4);

  }

  public List<String> getServerIpByWeight() {
      List<String> ips = new ArrayList<String>(32);
      for (Map.Entry<String, Integer> entry : ipMap.entrySet()) {
          String ip = entry.getKey();
          Integer weight = entry.getValue();
          // 根据权重不同,放入list 中的数量等同于权重,轮询出的的次数等同于权重
          for (int ipCount =0; ipCount < weight; ipCount++) {
              ips.add(ip);
          }
      }
      return ips;
  }

  public String randomWeightRobin(){
      List<String> ips = this.getServerIpByWeight();
      //循环随机数
      Random random=new Random();
      int index =random.nextInt(ips.size());
      String serverIp = ips.get(index);
      return  serverIp;
  }

  public static void main(String[] args) {
      TestRandomWeight testRandomWeightRobin=new TestRandomWeight();
      for (int i =0;i< 10 ;i++){
          String server= testRandomWeightRobin.randomWeightRobin();
          System.out.println(server);
      }
  }
}

输出结果:

192.168.1.3
192.168.1.3
192.168.1.2
192.168.1.1
192.168.1.2
192.168.1.1
192.168.1.3
192.168.1.2
192.168.1.2
192.168.1.3

 

随机法

负载均衡方法随机的把负载分配到各个可用的服务器上,通过随机数生成算法选取一个服务器,这种实现算法最简单,随之调用次数增大,这种算法可以达到每台服务器的请求量接近于平均。

package com.monkeyjava.learn.basic.robin;

import com.google.common.collect.Lists;

import java.util.List;
import java.util.Random;

public class TestRandom {


  private List<String> ips = Lists.newArrayList("192.168.1.1", "192.168.1.2", "192.168.1.3");


  public String randomRobin(){
      //随机数
      Random random=new Random();
      int index =random.nextInt(ips.size());
      String serverIp= ips.get(index);
      return  serverIp;

  }

  public static void main(String[] args) {
      TestRandom testRandomdRobin =new TestRandom();
      for (int i=0;i< 10 ;i++){
          String serverIp= testRandomdRobin.randomRobin();
          System.out.println(serverIp);
      }
  }
}

输出

192.168.1.3
192.168.1.3
192.168.1.1
192.168.1.2
192.168.1.1
192.168.1.3
192.168.1.2
192.168.1.3
192.168.1.3
192.168.1.2

 

IP_Hash算法

hash(ip)%N算法,通过一种散列算法把客户端来源IP根据散列取模算法将请求分配到不同的服务器上

优点:保证了相同客户端IP地址将会被哈希到同一台后端服务器,直到后端服务器列表变更。根据此特性可以在服务消费者与服务提供者之间建立有状态的session会话

缺点: 如果服务器进行了下线操作,源IP路由的服务器IP就会变成另外一台,如果服务器没有做session 共享话,会造成session丢失。

package com.monkeyjava.learn.basic.robin;

import com.google.common.collect.Lists;

import java.util.List;

public class TestIpHash {


  private List<String> ips = Lists.newArrayList("192.168.1.1", "192.168.1.2", "192.168.1.3");


  public String ipHashRobin(String clientIp){
      int hashCode=clientIp.hashCode();
      int serverListsize=ips.size();
      int index = hashCode%serverListsize;
      String serverIp= ips.get(index);
      return  serverIp;

  }

  public static void main(String[] args) {
      TestIpHash testIpHash =new TestIpHash();
      String servername= testIpHash.ipHashRobin("192.168.88.2");
      System.out.println(servername);
  }
}

输出结果

192.168.1.3

每次运行结果都一样

 

总结

到此这篇关于使用Java实现5种负载均衡算法的文章就介绍到这了,更多相关Java负载均衡算法内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://mp.weixin.qq.com/s/BTkb2YN3RvEtBV-I8htWEw

延伸 · 阅读

精彩推荐
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7482021-02-04
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30