负载均衡算法
负载均衡算法是一种用于分配网络流量的技术,以确保服务器和其他计算资源在负载高峰期间不会过度负担。以下是几种常见的负载均衡算法:
轮询(Round Robin):将请求依次分配给每个服务器,循环执行。
最少连接数(Least Connections):将请求发送到当前连接数最少的服务器上。
IP散列(IP Hashing):根据客户端IP地址对服务器进行散列,使得同一个客户端总是被分配到同一个服务器上。
加权轮询(Weighted Round Robin):为每个服务器设置一个权重值,按照权重比例来分配请求。
动态加权轮询(Dynamic Weighted Round Robin):根据实时监测到的各台机器处理能力动态调整其权重值,并采用加权轮询方式进行流量调度。
随机:随机的把负载分配到各个可用的服务器上,通过随机数生成算法选取一个服务器
负载均衡的优势:
- 减少停机
- 可扩展性
- 冗余
- 灵活性
- 高效
Java实现5种负载均衡算法
1. 轮询算法
import com.google.common.collect.Lists;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
/**
* 轮询算法
* 轮询算法按顺序把每个新的连接请求分配给下一个服务器,最终把所有请求平分给所有的服务器。
* 优点:绝对公平
* 缺点:无法根据服务器性能去分配,无法合理利用服务器资源。
* @author:
* @date: 2022-08-10 17:53
*/
public class Round<T> {
private AtomicInteger index = new AtomicInteger(0);
public <T> T roundRobin(List<T> data) {
T t ;
if (index.get() >= data.size()) {
index = new AtomicInteger(0);
}
t = data.get(index.get());
//轮询+1
index.getAndIncrement();
return t;
}
public static void main(String[] args) {
List<String> ips = Lists.newArrayList("192.168.1.1", "192.168.1.2", "192.168.1.3");
Round<String> testRoundRobin =new Round<String>();
for (int i=0;i< 10 ;i++){
String s = testRoundRobin.roundRobin(ips);
System.out.println(s);
}
}
}
2. 加权轮询法
import com.collmall.shortlink.entity.DbConfig;
import org.springframework.util.CollectionUtils;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.atomic.AtomicInteger;
/**
* 加权轮询法
* 该算法中,每个机器接受的连接数量是按权重比例分配的。这是对普通轮询算法的改进,
* 比如你可以设定:第三台机器的处理能力是第一台机器的两倍,那么负载均衡器会把两倍的连接数量分配给第3台机器,轮询可以将请求顺序按照权重分配到后端。
* @author
* @date 2022年08月11日14:46:25
*/
public class Weight<T> {
private AtomicInteger index = new AtomicInteger(0);
public List<T> getDataByWeight(List<T> data) throws NoSuchFieldException, IllegalAccessException {
List<T> ips = new CopyOnWriteArrayList<T>();
if (CollectionUtils.isEmpty(data)) {
return data;
}
for (T t : data) {
Field nameField = t.getClass().getDeclaredField("name");
// setAccessible 实体中的属性是用private定义的,需要设置setAccessible 为true,才可以访问到对象
nameField.setAccessible(true);
// 获取属性值
String name =(String) nameField.get(t);
Field weightField = t.getClass().getDeclaredField("weight");
// setAccessible 实体中的属性是用private定义的,需要设置setAccessible 为true,才可以访问到对象
weightField.setAccessible(true);
// 获取属性值
Integer weight = (Integer) weightField.get(t);
for (int ipCount =0; ipCount < weight; ipCount++) {
ips.add(t);
}
}
return ips;
}
public T weightRobin(List<T> data) throws NoSuchFieldException, IllegalAccessException {
List<T> list = this.getDataByWeight(data);
if (index.get() >= list.size()){
index = new AtomicInteger(0);
}
T t = list.get(index.get());
index.getAndIncrement();
return t;
}
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
Weight testWeightRobin=new Weight();
List<DbConfig> dbConfigList= new ArrayList<>();
dbConfigList.add(new DbConfig("192.168.1.1", 1));
dbConfigList.add(new DbConfig("192.168.1.2", 2));
dbConfigList.add(new DbConfig("192.168.1.3", 4));
for (int i =0;i< 10 ;i++){
DbConfig server = (DbConfig)testWeightRobin.weightRobin(dbConfigList);
System.out.println(server);
}
}
}
3. 加权随机法
import com.collmall.shortlink.entity.DbConfig;
import java.lang.reflect.Field;
import java.util.*;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ThreadLocalRandom;
/**
* 加权随机法
* 获取带有权重的随机数字,随机这种东西,不能看绝对,只能看相对,
* 我们不用index 控制下标进行轮询,只用random 进行随机取ip,即实现算法。
* @author
* @date 2022年08月14日11:37:20
*/
public class RandomWeight<T> {
public <T> List<T> getDataByWeight(List<T> data) throws NoSuchFieldException, IllegalAccessException {
List<T> ips = new CopyOnWriteArrayList<T>();
for (T t : data) {
Field nameField = t.getClass().getDeclaredField("name");
// setAccessible 实体中的属性是用private定义的,需要设置setAccessible 为true,才可以访问到对象
nameField.setAccessible(true);
// 获取属性值
String name =(String) nameField.get(t);
Field weightField = t.getClass().getDeclaredField("weight");
// setAccessible 实体中的属性是用private定义的,需要设置setAccessible 为true,才可以访问到对象
weightField.setAccessible(true);
// 获取属性值
Integer weight = (Integer) weightField.get(t);
// 根据权重不同,放入list 中的数量等同于权重,轮询出的的次数等同于权重
for (int ipCount =0; ipCount < weight; ipCount++) {
ips.add(t);
}
}
return ips;
}
public <T> T randomWeightRobin(List<T> data) throws NoSuchFieldException, IllegalAccessException {
List<T> ips = this.getDataByWeight(data);
// //循环随机数
// Random random=new Random();
// int index =random.nextInt();
int index = ThreadLocalRandom.current().nextInt(ips.size());
T t = ips.get(index);
return t;
}
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
RandomWeight<DbConfig> randomWeightRobin =new RandomWeight<DbConfig>();
List<DbConfig> dbConfigList= new ArrayList<>();
dbConfigList.add(new DbConfig("192.168.1.1", 1));
dbConfigList.add(new DbConfig("192.168.1.2", 2));
dbConfigList.add(new DbConfig("192.168.1.3", 4));
for (int i =0;i< 10 ;i++){
DbConfig server = randomWeightRobin.randomWeightRobin(dbConfigList);
System.out.println(server);
}
}
}
4. 随机法
import com.collmall.shortlink.entity.DbConfig;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ThreadLocalRandom;
/**
* 随机法
* 负载均衡方法随机的把负载分配到各个可用的服务器上,通过随机数生成算法选取一个服务器,
* 这种实现算法最简单,随之调用次数增大,这种算法可以达到每台服务器的请求量接近于平均。
* @author
* @date 2022年10月14日13:46:24
*/
public class Random<T> {
public <T> T randomRobin(List<T> data){
int index = ThreadLocalRandom.current().nextInt(data.size());
T t = data.get(index);
return t;
}
public static void main(String[] args) {
Random<DbConfig> randomdRobin =new Random<DbConfig>();
List<DbConfig> dbConfigList= new ArrayList<>();
dbConfigList.add(new DbConfig("192.168.1.1", 1));
dbConfigList.add(new DbConfig("192.168.1.2", 2));
dbConfigList.add(new DbConfig("192.168.1.3", 4));
for (int i=0;i< 10 ;i++){
DbConfig dbConfig = randomdRobin.randomRobin(dbConfigList);
System.out.println(dbConfig);
}
}
}
5. IP_Hash 法
import com.collmall.shortlink.entity.DbConfig;
import java.util.ArrayList;
import java.util.List;
/**
* IP_Hash算法
* hash(ip)%N算法,通过一种散列算法把客户端来源IP根据散列取模算法将请求分配到不同的服务器上
*
* 优点:保证了相同客户端IP地址将会被哈希到同一台后端服务器,直到后端服务器列表变更。根据此特性可以在服务消费者与服务提供者之间建立有状态的session会话
*
* 缺点: 如果服务器进行了下线操作,源IP路由的服务器IP就会变成另外一台,如果服务器没有做session 共享话,会造成session丢失。
* @author
* @date 2022年08月14日14:06:41
*/
public class IpHash<T> {
/**
*
* @param data
* @param hashData
* @param <T>
* @return
*/
public <T> T ipHashRobin(List<T> data,String hashData) {
// int hashCode = hashData.hashCode();
// int listSize = data.size();
// int index = hashCode % listSize;
int index = hashData.hashCode() % data.size();
T t = data.get(index);
return t;
}
public static void main(String[] args) {
IpHash<DbConfig> ipHash =new IpHash<DbConfig>();
List<DbConfig> dbConfigList= new ArrayList<>();
dbConfigList.add(new DbConfig("192.168.1.1", 1));
dbConfigList.add(new DbConfig("192.168.1.2", 2));
dbConfigList.add(new DbConfig("192.168.1.3", 4));
DbConfig dbConfig = ipHash.ipHashRobin(dbConfigList, "192.168.88.2");
System.out.println(dbConfig.toString());
}
}
6. 附录 DbConfig
public class DbConfig {
//id
private Long id;
//数据库名称
private String name;
//负载策略:1轮询算法 ,2 加权轮询法,3 加权随机法,4 随机法,5 IP_Hash算法
private Integer loadPolicy;
//权重: 只有 2 加权轮询法,3 加权随机法 才会用到
private Integer weight;
//是否删除:0 未删除,1是已删除
private Integer isDeleted;
//创建时间
private Date cT;
//更新时间
private Date uT;
public DbConfig(String name, Integer weight) {
this.name = name;
this.weight = weight;
}
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getLoadPolicy() {
return loadPolicy;
}
public void setLoadPolicy(Integer loadPolicy) {
this.loadPolicy = loadPolicy;
}
public Integer getWeight() {
return weight;
}
public void setWeight(Integer weight) {
this.weight = weight;
}
public Integer getIsDeleted() {
return isDeleted;
}
public void setIsDeleted(Integer isDeleted) {
this.isDeleted = isDeleted;
}
public Date getCT() {
return cT;
}
public void setCT(Date cT) {
this.cT = cT;
}
public Date getUT() {
return uT;
}
public void setUT(Date uT) {
this.uT = uT;
}
/**
* 获取主键值
*
* @return 主键值
*/
@Override
protected Serializable pkVal() {
return this.id;
}
@Override
public String toString() {
return "DbConfig{" +
"id=" + id +
", name='" + name + '\'' +
", loadPolicy=" + loadPolicy +
", weight=" + weight +
", isDeleted=" + isDeleted +
", cT=" + cT +
", uT=" + uT +
'}';
}
}