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;不要给自己设限。为什么是每天一…

设计模式 工厂方法_工厂设计模式–一种有效的方法

设计模式 工厂方法如您所知&#xff0c;“工厂方法模式”或俗称“工厂设计模式”是“创意设计模式”类别下的一种设计模式。 模式背后的基本原理是&#xff0c;在运行时&#xff0c;我们根据传递的参数获得类似类型的对象。 关于这种模式的文章很多&#xff0c;开发人员可以通过…

SMPP Java示例(客户端)

这篇文章通过创建一个简单的SMPP客户端向移动用户发送短信来提供SMPP Java示例&#xff0c;使用该客户端我们可以简单地提交以向单个移动用户发送消息&#xff0c;也可以一次将消息广播给多个移动用户。另外&#xff0c;我们将验证交货收据。 出于客户端的目的&#xff0c;我们…

mysql插入另一个表中数据_MySql中把一个表的数据插入到另一个表中的实现

1.如果2张表的字段一致&#xff0c;并且希望插入全部数据&#xff0c;可以用这种方法&#xff1a;INSERT INTO 目标表 SELECT * FROM 来源表;insert into insertTest select * from insertTest2;2.如果只希望导入指定字段&#xff0c;可以用这种方法&#xff1a;INSERT INTO 目…

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

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

mysql自定义函数多参数_自定义mysql函数 - 无法传递参数

作为标题状态。这里的功能DELIMITER //CREATE FUNCTION GetCreateValue( table_name CHAR(64), id_field CHAR(64), name_field CHAR(64), name_value CHAR(64) )RETURNS INTEGERBEGINDECLARE ret INTEGER;SELECT count(*) INTO ret FROM table_name WHERE name_field name_va…

与Zapier集成

整合很无聊。 也是不可避免的。 但是我不会写有关企业集成模式的文章 。 相反&#xff0c;我将解释如何创建与Zapier集成的应用程序。 什么是Zapier &#xff1f; 它是一项服务&#xff0c;使您可以通过其API&#xff08;或协议&#xff09;连接两个&#xff08;或多个&#x…

python用psf函数_python学习之-类的内置函数

内置方法&#xff1a;__str__(该方法必须返回字符串类型),在对像被打印时自动触发,然后将该方法的返回值当做打印结果输出)class People:def __init__(self,name,age):self.namenameself.ageagedef __str__(self): #绑定给对象的方法return ‘‘ %(self.name,self.age) #这个方…

多用户远程连接mysql_Mysql权限控制 - 允许用户远程连接

Mysql为了安全性&#xff0c;在默认情况下用户只允许在本地登录&#xff0c;可是在有此情况下&#xff0c;还是需要使用用户进行远程连接&#xff0c;因此为了使其可以远程需要进行如下操作&#xff1a;一、允许root用户在任何地方进行远程登录&#xff0c;并具有所有库任何操作…

cxf 服务端soap报文_使用Apache CXF开发SOAP Web服务

cxf 服务端soap报文在上一篇文章中&#xff0c;我逐步介绍了使用apache CXF开发简单的RESTFull服务的步骤。 在本文中&#xff0c;我将讨论使用CXF开发SOAP Web服务。 在继续前进之前&#xff0c;让我们先了解一下构成SOAP Web服务的概念/元素 SOAP或简单对象访问协议 SOAP是一…

Java 8日期和时间

如今&#xff0c;一些应用程序仍在使用java.util.Date和java.util.Calendar API&#xff0c;包括使我们的生活更轻松地使用这些类型的库&#xff0c;例如JodaTime。 但是&#xff0c;Java 8引入了新的API来处理日期和时间&#xff0c;这使我们可以对日期和时间表示进行更精细的…

php mysql 随机字符串_MySQL_Mysql 自定义随机字符串的实现方法,前几天在开发一个系统,需要 - phpStudy...

Mysql 自定义随机字符串的实现方法前几天在开发一个系统&#xff0c;需要用到随机字符串&#xff0c;但是mysql的库函数有没有直接提供&#xff0c;就简单的利用现有的函数东拼西凑出随机字符串来.下面简单的说下实现当时.1.简单粗暴.select ..., substring(MD5(RAND()),floor(…

python哪些类型可以作为迭代器_python0.8-----set类型与迭代器

set:类似dict&#xff0c;是一组dict的集合&#xff0c;不存储value。本质&#xff1a;无序(没有下标)无重复的元素的集合。创建&#xff1a;创建set需要一个list或者tuple或者dict作为输入集合。重复元素在色中会被自动过滤。s1[1,2,3,4,5,6,1,2]set(s1){1,2,3,4,5,6} --表示》…

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

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

python线性回归x可以数量不一样吗_R和Python中的线性回归 - 在同一问题上的结果不同...

只是指出这一点&#xff1a; statsmodel s least squares fit does by default not include a constant. If we remove the constant from R适合&#xff0c;我们得到与Python实现非常相似的结果&#xff0c;或者相反&#xff0c;如果我们向 statsmodel -fit添加一个常量&#…

PostgreSQL开放自由

Open Liberty是源自WebSphere Liberty的有趣的新型OSS Java EE应用服务器。 您可以将Open Liberty配置为使用PostgreSQL作为其默认数据源&#xff0c;如下所示&#xff1a; 将<datasource> &#xff0c; <jdbcDriver>和<library>指令添加到server.xml配置中…

实现mysql百度式查询_mysql查询优化建议(百度)

1.对查询进行优化&#xff0c;应尽量避免全表扫描&#xff0c;首先应考虑在 where 及 order by 涉及的列上建立索引。2.应尽量避免在 where 子句中使用!或<>操作符&#xff0c;否则将引擎放弃使用索引而进行全表扫描。3.应尽量避免在 where 子句中对字段进行 null 值判断…

海思芯片怎么使用tde给qt加速_3519移植Qt适配附件

【实例简介】修改Qt5.5.1linuxfb插件代码&#xff0c;适配海思3519平台【实例截图】【核心代码】3519Qt移植.tar├── 3519Qt绉绘│ ├── linuxfb│ │ ├── include│ │ │ ├── acodec.h│ │ │ ├── fisheye_calibrate.h│ │ │ ├─…