mysql 一致性hash_一致性hash算法在分表分库中的应用

32b183f7200afca61cc74fb6c447274c.png 一致性hash算法结构图

7a10455fa13593c51678d8065a3e3157.png 分表分库结构图

可进行循环冗余存储,顺时针存储到下一个物理节点(非虚拟节点)

package com.haiziwang.platform.kmcsms.route.algorithm;

import java.util.Collection;

import java.util.SortedMap;

import java.util.TreeMap;

/**

* 一致性Hash算法

* 算法详解:http://blog.csdn.net/sparkliang/article/details/5279393

* 算法实现:https://weblogs.java.net/blog/2007/11/27/consistent-hashing

*

* @param 节点类型,比如ip或host等

* @author xiaoleilu

*/

public class ConsistentHash {

/**

* Hash计算对象,用于自定义hash算法

*/

HashFunc hashFunc;

/**

* 复制的节点个数

*/

private final int numberOfReplicas;

/**

* 一致性Hash环

*/

private final SortedMap circle = new TreeMap();

/**

* 构造,使用Java默认的Hash算法

*

* @param numberOfReplicas 复制的节点个数,增加每个节点的复制节点有利于负载均衡

* @param nodes 节点对象

*/

public ConsistentHash(int numberOfReplicas, Collection nodes) {

this.numberOfReplicas = numberOfReplicas;

this.hashFunc = new HashFunc() {

@Override

public Integer hash(Object key) {

String data = key.toString();

//默认使用FNV1hash算法

final int p = 16777619;

int hash = (int) 2166136261L;

for (int i = 0; i < data.length(); i++)

hash = (hash ^ data.charAt(i)) * p;

hash += hash << 13;

hash ^= hash >> 7;

hash += hash << 3;

hash ^= hash >> 17;

hash += hash << 5;

return hash;

}

};

//初始化节点

for (T node : nodes) {

add(node);

}

}

/**

* 构造

*

* @param hashFunc hash算法对象

* @param numberOfReplicas 复制的节点个数,增加每个节点的复制节点有利于负载均衡

* @param nodes 节点对象

*/

public ConsistentHash(HashFunc hashFunc, int numberOfReplicas, Collection nodes) {

this.numberOfReplicas = numberOfReplicas;

this.hashFunc = hashFunc;

//初始化节点

for (T node : nodes) {

add(node);

}

}

/**

* 增加节点

* 每增加一个节点,就会在闭环上增加给定复制节点数

* 例如复制节点数是2,则每调用此方法一次,增加两个虚拟节点,这两个节点指向同一Node

* 由于hash算法会调用node的toString方法,故按照toString去重

*

* @param node 节点对象

*/

public void add(T node) {

for (int i = 0; i < numberOfReplicas; i++) {

circle.put(hashFunc.hash(node.toString() + i), node);

}

}

/**

* 移除节点的同时移除相应的虚拟节点

*

* @param node 节点对象

*/

public void remove(T node) {

for (int i = 0; i < numberOfReplicas; i++) {

circle.remove(hashFunc.hash(node.toString() + i));

}

}

/**

* 获得一个最近的顺时针节点

*

* @param key 为给定键取Hash,取得顺时针方向上最近的一个虚拟节点对应的实际节点

* @return 节点对象

*/

public T get(Object key) {

if (circle.isEmpty()) {

return null;

}

int hash = hashFunc.hash(key);

if (!circle.containsKey(hash)) {

SortedMap tailMap = circle.tailMap(hash); //返回此映射的部分视图,其键大于等于 hash

hash = tailMap.isEmpty() ? circle.firstKey() : tailMap.firstKey();

}

//正好命中

return circle.get(hash);

}

/**

* Hash算法对象,用于自定义hash算法

*

* @author xiaoleilu

*/

public interface HashFunc {

public Integer hash(Object key);

}

}

package com.haiziwang.platform.kmcsms.route.demo;

/**

* Created by wangzhongbao on 2016/12/14.

*/

public class NodeBean {

private String dbIp;

public NodeBean(String dbIp) {

this.dbIp = dbIp;

}

public String getDbIp() {

return dbIp;

}

@Override

public String toString() {

return dbIp;

}

/*@Override

public boolean equals(Object obj) {

return this.toString().equals(obj.toString());

}

@Override

public int hashCode() {

return 1;

}*/

}

package com.haiziwang.platform.kmcsms.route.demo;

/**

* Created by wangzhongbao on 2016/12/14.

*/

public class TelUtil {

private static String[] telFirst="134,135,136,137,138,139,150,151,152,157,158,159,130,131,132,155,156,133,153".split(",");

public static String getTel() {

int index= getNum(0,telFirst.length-1);

String first=telFirst[index];

String second=String.valueOf(getNum(1,10000)+10000).substring(1);

String thrid=String.valueOf(getNum(1,10000)+10000).substring(1);

return first+second+thrid;

}

private static int getNum(int start,int end) {

return (int)(Math.random()*(end-start+1)+start);

}

}

package com.haiziwang.platform.kmcsms.route.demo;

import com.haiziwang.platform.kmcsms.route.algorithm.ConsistentHash;

import java.util.ArrayList;

import java.util.HashMap;

import java.util.List;

import java.util.Map;

/**

* Created by wangzhongbao on 2016/12/14.

*/

public class Sample {

public static void main(String[] args) {

//每个节点虚拟出来的节点数

final int numberOfReplicas = 10000;

List nodeList = new ArrayList();

nodeList.add(new NodeBean("000"));

nodeList.add(new NodeBean("111"));

nodeList.add(new NodeBean("222"));

nodeList.add(new NodeBean("333"));

nodeList.add(new NodeBean("444"));

nodeList.add(new NodeBean("555"));

nodeList.add(new NodeBean("666"));

nodeList.add(new NodeBean("777"));

nodeList.add(new NodeBean("888"));

nodeList.add(new NodeBean("999"));

ConsistentHash stringConsistentHash = new ConsistentHash(numberOfReplicas,nodeList);

Map countMap = new HashMap(){

@Override

public Integer put(NodeBean key, Integer value) {

Integer oldVal = this.get(key);

if(oldVal==null){

return super.put(key, value);

}else{

return super.put(key,oldVal+1);

}

}

};

long l = System.currentTimeMillis();

for(int i=0;i<1000000;i++){

NodeBean nodeBean = stringConsistentHash.get(TelUtil.getTel());

countMap.put(nodeBean, 1);

}

System.out.println("1000000计算耗时:"+(System.currentTimeMillis() - l)+"毫秒");

for(NodeBean nb:countMap.keySet()){

System.out.println(nb.toString()+" 分配到个数:"+countMap.get(nb));

}

}

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/348072.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

生产中的性能分析

生产中的性能分析 如果您在Java应用程序的性能方面遇到了一些严重问题&#xff0c;那么很可能您会知道线程分析的价值。 但是您知道应该使用哪个分析器吗&#xff1f; 探查器使用两种基本技术-采样和仪器。 采样分析器 采样探查器包括定期向JVM询问所有当前活动线程的当前执行…

mysql用com_MySQL 使用教程

关于 MySQLMySQL 是最流行的开源数据库。本文简明的讲解了 MySQL 如何下载安装到使用的整个过程。MySQL 支持多种特性&#xff1a;使用 C和 C编写&#xff0c;并使用了多种编译器进行测试&#xff0c;保证了源代码的可移植性。支持 AIX、FreeBSD、HP-UX、Linux、Mac OS、Novell…

c++ 一维数组长度_每天一点C / 一维数组和指针

哈喽&#xff0c;我是老吴&#xff0c;继续记录我的学习心得。每天一点系列是我对微习惯的践行。现在能做到每天一点 C&#xff0c;将来就会有更多的每天一点系列&#xff0c;没人规定嵌入式软件工程师就只能学习 C 语言和折腾 Linux&#xff0c;不要给自己设限。为什么是每天一…

mysql事务中怎么更改空值_MySQL事务

1.事务1.事务特性--ACIDAtomicity(原子性):要么全做,要么不做,不能只做一半(银行转账)Consistency(约束性):事务的前后,约束都能满足Isolation(依赖性):事务之间是独立的,互不影响的Durability(持久性):事务执行之后,事物的结果可以持久保存2.事务隔离级别:read uncommitted:可…

线程池实现填充短信_填充一个池需要多少个线程?

线程池实现填充短信在最近几个月中&#xff0c;我们一直看到很小但持续的操作失败&#xff0c;但有一个奇怪的异常– org.springframework.jdbc.CannotGetJdbcConnectionException –“无法获得JDBC连接&#xff1b; 嵌套异常是java.sql.SQLException&#xff1a;客户端尝试检出…

jsf标签_多个动态包含一个JSF标签

jsf标签每个JSF开发人员都知道ui&#xff1a;include和ui&#xff1a;param标签。 您可以包括一个facelet&#xff08;XHTML文件&#xff09;并传递一个对象&#xff0c;该对象将在包含的facelet中可用&#xff0c;如下所示&#xff1a; <ui:include src"/sections/co…

spring 长轮询_Spring集成文件轮询和测试

spring 长轮询我最近实施了一个小项目&#xff0c;在该项目中&#xff0c;我们必须轮询文件夹中的新文件&#xff0c;然后在文件内容上触发服务流。 Spring Integration非常适合此要求&#xff0c;因为它带有一个通道适配器 &#xff0c;该适配器可以扫描文件夹中的新文件&…

您将在下一个项目中使用JSF吗?

上周有一篇很棒的stackoverflow博客文章&#xff0c;主题是“ JavaScript框架的残酷生命周期” 。 这篇文章是关于Javascript UI框架&#xff08;angularjs&#xff0c;angular&#xff0c;jquery和react&#xff09;的流行和流行的速度。 这篇文章的关键指标是每月关于框架的问…

使用java自带的日志管理_java日志管理

1.相关概念日志统一框架(日志门面)&#xff1a;apache commons logging、slf4j日志实现框架(实现层)&#xff1a;JDK自带的logging(java.util.logging)、log4j、Java Util Logging、log4j2、logback.(1)JDK自带的logging(java.util.logging)用法&#xff1a;1 importjava.util.…

安卓4.4玩java_Android4.4运行过程中闪退java.lang.NoClassDefFoundError

上周五项目测试时发现一个奇怪的Bug&#xff0c;项目中依赖了一个第三方框架&#xff0c;但是在android4.0-4.4.4之间的系统中运行会直接闪退&#xff0c;抛出错误异常为java.lang.NoClassDefFoundError。第一次遇到这样的问题&#xff0c;google了好久找到了以下几个原因&…

update se_Java SE 7 Update 25 –发行说明进行了解释。

update se昨天是CPU日。 Oracle通过6月的Java重要补丁更新发布了Java SE更新25 。 在4月的最后一次重大更新之后&#xff0c;这是最后一次与Oracle其他所有Oracle产品都不适合的Oracle重要补丁更新计划。 从2013年10月开始 &#xff0c;Java安全修补程序将遵循四个年度安全发布…

java scavenge_请概述一下Java中都有哪些垃圾收集器

1、Serial(串行GC)收集器Serial收集器是一个新生代收集器&#xff0c;单线程执行&#xff0c;使用复制算法。它在进行垃圾收集时&#xff0c;必须暂停其他所有的工作线程(用户线程)。是Jvmclient模式下默认的新生代收集器。对于限定单个CPU的环境来说&#xff0c;Serial收集器由…

阿帕奇光束

Apache Beam是一个开放源代码统一模型&#xff0c;用于定义批处理和流数据并行处理管道。 使用一种开源的Beam SDK&#xff0c;您可以构建一个定义管道的程序。 然后&#xff0c;该管道由Beam支持的分布式处理后端之一执行&#xff0c;这些后端包括Apache Apex &#xff0c; Ap…

java音乐登陆界面_第四篇——Spring音乐登录界面设计及实现(C#)

Spring音乐播放器&#xff0c;我们小组设计其启动时有一个登录界面&#xff0c;用户初次可以注册&#xff0c;输入用户名和密码&#xff0c;点击注册即将输入信息保存到register文本文件中&#xff1b;如果已有用户名&#xff0c;输入用户名和密码&#xff0c;点击登录&#xf…

java中线性结构的例子_java数据结构--线性结构

一、数据结构数据结构由数据和结构两部分组成&#xff0c;就是将数据按照一定的结构组合起来&#xff0c;这样不同的组合方式有不同的效率&#xff0c;可根据需求选择不同的结构应用在相应在场景。数据结构大致分为两类&#xff1a;线性结构(如数组&#xff0c;链表&#xff0c…

Spring Boot和JSP

本指南展示了如何使用Spring Boot通过JSP创建MVC Web应用程序。 先决条件&#xff1a; Eclipse IDE&#xff08;最新版本&#xff09; Maven的4 Java 1.8 1.创建Maven Web项目 打开eclipse&#xff0c;然后创建一个新的Maven Web项目&#xff0c;并将其命名为SpringBootWe…

判断用户名和密码是否正确java_facebook中如何判断用户名和密码是否正确

Do you hear the angel?Do you think that this really has the angel in the world?We knew from the story in the book, the angel contain the wing of the beauty, beautiful face , the mind of the docile, generous breadth of view, they would like to sacrifice t…

使用Spock测试您的代码

Spock是针对Java和Groovy应用程序的测试和规范框架。 Spock是&#xff1a; 极富表现力 促进测试的给定/何时/然后语法 与大多数IDE和CI服务器兼容。 听起来不错&#xff1f; 通过快速访问Spock Web控制台&#xff0c;您可以非常快速地开始使用Spock。 当您有一个喜欢的小测…

php删除文见,php如何删除文件夹

php删除文件夹的方法&#xff1a;首先创建一个PHP示例文件&#xff1b;然后通过“function deldir($dir) {...}”删除目录下的文件并删除当前文件夹即可。php删除文件夹及其文件夹下所有文件function deldir($dir) {//先删除目录下的文件&#xff1a;$dhopendir($dir);while ($…

阿帕奇齿轮泵

Apache Gearpump是实时大数据流引擎。 它从2014年中期开始就在Intel上作为GitHub上的开源项目构想的&#xff0c;并于2016年3月8日进入Apache孵化。Gearpump的名称是对工程术语“齿轮泵”的指称&#xff0c;它是一个超级简单的泵&#xff0c;由只有两个齿轮&#xff0c;但是在流…