redis---一致性hash特性及java实现

1.几个特性

  • 单调性

单调性是指如果已经有一些内容通过哈希分派到了相应的缓冲中,又有新的缓冲区加入到系统中,那么哈希的结果应能够保证原有已分配的内容可以被映射到新的缓冲区中去,而不会被映射到旧的缓冲集合中的其他缓冲区。

有点绕,简单来说就是,如果添加一个新节点,原有的key对应的缓存,要么不迁移,要么迁移到新的节点,而不会迁移到其他旧节点。

  • 均衡性
    均衡性就是指hash的结果要尽可能地均分到所有的缓存节点中。
  • 分散性
    分散性是指同一个key对应的缓存仅放在一个节点上,每次访问也能访问到。而不用多个节点上保存相同的内容。

2.Java版本实现

public class LZHash {
    static List<String> realNodes = new ArrayList<String>();
    int replcaNum = 50;//虚拟节点副本数量,需要大一些,能更好地保证均衡性
    static SortedMap<Integer, String> virtualNodes = new TreeMap<Integer, String>();

    public LZHash(List<String> realNodes) {
        for(String node : realNodes) {
            addNode(node);
        }
    }

    private void addNode(String node) {
        for(int i = 0; i < replcaNum; i ++) {
            virtualNodes.put(md5(node + i).hashCode(), node);
        }

        realNodes.add(node);
    }

    private void removeNode(String node) {
        for(int i = 0; i < replcaNum; i++) {
            virtualNodes.remove(md5(node + i));
        }

        realNodes.remove(node);
    }

    public static String md5(String source) {
        return Hashing.md5().newHasher().putString(source, Charsets.UTF_8).hash().toString();
    }

    public static String getRedisNode(String key) {
        int hash = md5(key).hashCode();

        if(virtualNodes.containsKey(hash)) {
            return virtualNodes.get(hash);
        }

        SortedMap<Integer, String> temp = virtualNodes.tailMap(hash);
        if(temp.isEmpty()) {
            return virtualNodes.get(virtualNodes.firstKey());
        }

        return temp.get(temp.firstKey());

    }

    public static void main(String[] args) {
        List<String> realNodes = new ArrayList<String>();
        realNodes.add("192.168.96.220:6379");
        realNodes.add("192.168.99.233:6379");
        realNodes.add("192.168.99.60:6379");
        LZHash h = new LZHash(realNodes);

        String keyPrefix = "binova_shopcart_";
        int node220 = 0;
        int node233 = 0;
        int node60 = 0;
        for(int i = 0; i < 100000; i++) {
            String key = keyPrefix + i;
            String node = getRedisNode(key);
            if(node.equals("192.168.96.220:6379")) {
                node220 ++;
            }if(node.equals("192.168.99.233:6379")) {
                node233 ++;
            }if(node.equals("192.168.99.60:6379")) {
                node60 ++;
            }
        }
        System.out.println("node220:" + node220);
        System.out.println("node233:" + node233);
        System.out.println("node60:" + node60);
    }

}
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值