一致性哈希算法

博客介绍了分布式缓存中请求转发问题,传统做法在增加主机时会导致大部分缓存失效、引发缓存雪崩。为解决该问题,需保证取余部分不变,将请求转发到映射位置后的第一台主机。此外,还给出了一致性哈希算法的Java代码实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

介绍

在分布式缓存中,每台缓存主机中存放的缓存数据不同,因此在读取缓存数据时,应该将请求正确转发到对应的缓存服务器。传统的做法是先给目标主机编号,如0~(N-1),当收到请求时,计算请求标识符fid的哈希值fHash,之后将请求转发到fHash%N号的主机上。但当增加一台主机时,将会将请求转发到fHash%(N+1)的主机上,从而会使得大部分缓存失效,从而导致缓存雪崩。
为了解决该问题,只需要将取余部分不变,就能使得每次哈希计算结果不变。但是目标主机的哈希值可能和请求标识符的最终映射的位置不同,因此会转发到映射位置后的第一台主机上。

具体参考:原文链接

代码实现(Java)

生成唯一哈希值的接口及实现

public interface HashingFunc {
	public Long hash(Object key); 
}
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
public class ConcreteHashingFunc implements HashingFunc {
	@Override
	public Long hash(Object key) {
		return md5HashingAlg(key.toString());
	}
    /**
     * 使用MD5算法
     * @param key
     * @return
     */
    private long md5HashingAlg(String key) {
        MessageDigest md5 = null;
        try {
            md5 = MessageDigest.getInstance("MD5");
            md5.reset();
            md5.update(key.getBytes());
            byte[] bKey = md5.digest();
            long res = ((long) (bKey[3] & 0xFF) << 24) | ((long) (bKey[2] & 0xFF) << 16) | ((long) (bKey[1] & 0xFF) << 8)| (long) (bKey[0] & 0xFF);
            return res;
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        }
        return 0l;
    }
}

一致性哈希算法

import java.util.Collection;
import java.util.SortedMap;
import java.util.TreeMap;
/**
 * 一致性哈希函数,用来计算某个请求对应访问的主机号
 * @param <T>	主机类型
 */
public class ConsistenceHashing<T> {
	private HashingFunc hashingFunc;		//计算key的哈希值的接口
	private int copyNodeNum;				//每个结点对应的虚拟结点个数
	private SortedMap<Long, T> cicleMap = new TreeMap<Long , T>();
	/**
	 * 构造一致性Hashing对象
	 * @param hashingFunc	哈希函数
	 * @param copyNodeNum	每个结点对应的虚拟结点个数
	 * @param nodes			主机结点
	 */
	public ConsistenceHashing(HashingFunc hashingFunc,int copyNodeNum,Collection<T> nodes) {
		this.hashingFunc = hashingFunc;
		this.copyNodeNum = copyNodeNum;
		for (T t : nodes) {
			add(t);
		}
	}
	
	/**
	 * 增加主机结点
	 * @param node	主机结点
	 */
	public void add(T node){
		for (int i = 0; i < copyNodeNum; i++) {
			cicleMap.put(hashingFunc.hash(node.toString() + i), node);
		}
	}
	
	/**
	 * 获取当前请求对应的主机号
	 * @param kid	请求标识
	 * @return	
	 */
	public T get(Object kid){
		if(cicleMap.isEmpty()){
			return null;
		}
		Long kHash = hashingFunc.hash(kid);
		SortedMap<Long, T> tailMap = cicleMap.tailMap(kHash);		//获取key值大于等于kHash的部分	
		kHash = tailMap.isEmpty()?cicleMap.firstKey() : tailMap.firstKey();
		return cicleMap.get(kHash);
	}
	
	/**
	 * 移除目标主机
	 * @param node	主机结点
	 */
	public void remove(T node){
		for (int i = 0; i < copyNodeNum; i++) {
			cicleMap.remove(hashingFunc.hash(node.toString() + i));
		}
	}
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值