【JavaEE】自主设计实现负载均衡框架

首先来看看百度百科对于负载均衡的描述:
负载均衡建立在现有网络结构之上,它提供了一种廉价有效透明的方法扩展网络设备和服务器的带宽、增加吞吐量、加强网络数据处理能力、提高网络的灵活性和可用性。
负载均衡(Load Balance)其意思就是分摊到多个操作单元上进行执行,例如Web服务器、FTP服务器、企业关键应用服务器和其它关键任务服务器等,从而共同完成工作任务。(负载均衡)
我对于负载均衡的理解是,因为网络的一些原因,会使得众多服务器种某个服务器压力过于大,导致传输缓慢,我们可以提供一种方法,去均衡一下各个服务器之间的关系,使他们能够均衡的去完成网络传输,而不至于因为某些服务器的崩溃,使得整个系统变得延迟缓慢。下面,我们来进行一下负载均衡框架的实现,这将会对我以后的项目等产生很大的影响。

为了方便理解,先画个图表示:
【JavaEE】自主设计实现负载均衡框架_第1张图片图中可以看到客户端有很多的服务器可以选择,那如何进行选择,能保证各个服务器的连接数量相对均衡,就需要实现负载均衡。
这里有两种实现思路:
一: 将所有的服务器形成一条循环链,增加的时候,在链的末尾增加,删除的时候,将指定的服务器进行删除即可,取得一个服务器的时候,通过给一个currentNode指针来对这条链进行轮询,如果当前只有一个服务器,那就是这个服务器,如果有多个服务器,那就进行遍历,保证每次将这些服务器均匀取得。
来看具体代码:
首先我们需要一个INetNode来取得服务器的ip和port:

public interface INetNode {
	String getIp();
	int getPort();
}

public class DefaultNetNode implements INetNode {
	private String ip;
	private int port;

	public DefaultNetNode() {
	}
	
	public DefaultNetNode(String ip, int port) {
		this.ip = ip;
		this.port = port;
	}

	public void setIp(String ip) {
		this.ip = ip;
	}

	public void setPort(int port) {
		this.port = port;
	}

	@Override
	public String getIp() {
		return ip;
	}

	@Override
	public int getPort() {
		return port;
	}

	@Override
	public int hashCode() {
		final int prime = 31;
		int result = 1;
		result = prime * result + ((ip == null) ? 0 : ip.hashCode());
		result = prime * result + port;
		return result;
	}

	@Override
	public boolean equals(Object obj) {
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (!(obj instanceof DefaultNetNode))
			return false;
		DefaultNetNode other = (DefaultNetNode) obj;
		if (ip == null) {
			if (other.ip != null)
				return false;
		} else if (!ip.equals(other.ip))
			return false;
		if (port != other.port)
			return false;
		return true;
	}

	@Override
	public String toString() {
		return "[" + ip + ":" + port + "]";
	}

}

需要定义该框架的功能:增加,删除,取得服务器:

public interface INetNodeBalance {
	void addNode(INetNode node);
	INetNode removeNode(INetNode node);
	INetNode getNode();
}

针对这个接口,我们需要进行扩展,给一个map来存放所有的服务器:

public abstract class AbstractNetNodeBalance implements INetNodeBalance {
	protected final Map<Integer, INetNode> nodePool;

	public AbstractNetNodeBalance() {
		this.nodePool = new ConcurrentHashMap<>();
	}
	
	@Override
	public void addNode(INetNode node) {
		if (node == null) {
			return;
		}
		
		int nodeHashCode = node.hashCode();
		if (nodePool.containsKey(nodeHashCode)) {
			return;
		}
		
		nodePool.put(nodeHashCode, node);
	}
	
	public boolean isNodePoolEmpty() {
		return nodePool.isEmpty();
	}
	
	@Override
	public INetNode removeNode(INetNode node) {
		if (node == null) {
			return null;
		}
		return nodePool.remove(node.hashCode());
	}
	
}

可以发现,这里用到了抽象类,为什么要给抽象类呢?很简单,因为我们接口定义了三个方法,而在AbstractNetNodeBalance 类中,只实现了两个,get方法还没有给出,因为get方法需要进行不同的负载均衡,所以,他应该由他的子类进行实现,不同的子类,它的负载均衡方法是不一样的。
下面看我的轮询方式:

public class PollingNetNode extends DefaultNetNode {
	private PollingNetNode next;
	private PollingNetNode pre;
	
	public PollingNetNode() {
		this.next = this;
		this.pre = this;
	}

	public PollingNetNode(String ip, int port) {
		super(ip, port);
		this.next = this;
		this.pre = this;
	}

	public void setNext(PollingNetNode next) {
		this.next = next;
	}

	public void setPre(PollingNetNode pre) {
		this.pre = pre;
	}

	public PollingNetNode getNext() {
		return this.next;
	}
	
	public PollingNetNode getPre() {
		return this.pre;
	}
	
}

public class PollingBalance  extends AbstractNetNodeBalance {
	private PollingNetNode pollingNode;
	private PollingNetNode currentNode;

	public PollingBalance() {
		super();
	}
	
	@Override
	public synchronized void addNode(INetNode node) {
		PollingNetNode newNode = new PollingNetNode(node.getIp(), node.getPort());
		
		if (pollingNode == null) {
			pollingNode = newNode;
			currentNode = pollingNode;
			super.addNode(newNode);
			return;
		}
		newNode.setPre(pollingNode.getPre());
		newNode.setNext(pollingNode);
		pollingNode.setPre(newNode);
		newNode.getPre().setNext(newNode);
		
		super.addNode(newNode);
	}

	@Override
	public synchronized INetNode removeNode(INetNode node) {
		PollingNetNode target = new PollingNetNode(node.getIp(), node.getPort());
		target = (PollingNetNode) super.removeNode(target);
		
		if (target == null) {
			return null;
		}
		
		if (isNodePoolEmpty()) {
			pollingNode = null;
			currentNode = null;
			return pollingNode;
		}
		
		if (currentNode == target) {
			currentNode = target.getNext();
		}
		
		if (pollingNode == target) {
			pollingNode = target.getNext();
		}
		
		target.getPre().setNext(target.getNext());
		target.getNext().setPre(target.getPre());
		target.setPre(target);
		target.setNext(target);
		
		return target;
	}

	@Override
	public synchronized INetNode getNode() {
		INetNode result = currentNode;
		
		if (currentNode != null) {
			currentNode = currentNode.getNext();
		}
		
		return result;
	}

}

对于上述的轮询,可以画个图来进行表示:
【JavaEE】自主设计实现负载均衡框架_第2张图片第二种实现思路,通过随机数的方式,也就是,产生一个随机数,对服务器的总个数进行取余,获得的那个下标就是所取的服务器。

public class RandomBalance extends AbstractNetNodeBalance {
	private final List<INetNode> nodeList;

	public RandomBalance() {
		super();
		nodeList = new LinkedList<>();
	}

	@Override
	public void addNode(INetNode node) {
		super.addNode(node);
		if (node == null || nodeList.contains(node)) {
			return;
		}
		nodeList.add(node);
	}

	@Override
	public INetNode removeNode(INetNode node) {
		if (node == null || !nodeList.contains(node)) {
			return null;
		}
		nodeList.remove(node);
		return super.removeNode(node);
	}

	@Override
	public INetNode getNode() {
		if (isNodePoolEmpty()) {
			return null;
		}
		
		int index = ((int) (Math.random() * 10000)) % nodeList.size();
		return this.nodeList.get(index);
	}

}

上述是我自己实现的两个负载均衡思路,并且写成了接口,以后,我们想要进行负载均衡的时候,只需要提供一个set方法,set接口的实现类进去就可以直接调用负载均衡了。

你可能感兴趣的:(负载均衡,自己实现)