赞
踩
字符串Hash算法指的是,从依据字段的内容中,截取字符串中指定位置的子字符串,然后将这个子字符串进行Hash运算,得到一个二进制数字,然后再拿这个二进制数字与1023也就是1111111111进行位运算,最后计算出该条数据应该落在哪个分片上。
字符串Hash算法与固定Hash算法类似,只不过固定Hash算法不能针对字符串处理,而字符串Hash算法可以将字符串转换成二进制数,然后再通过固定Hash算法计算出数据要落在哪个分片上。
如下图所示,依据字段列的值为tomcat,我们截取长度为0:1,截取两位,也就是to,此时字符串Hash会计算对to进行计算,得出一个二进制数字,然后再将这个二进制数据与1111111111进行位运算,根据得出的结果找到对应的分片ID,最终将数据写入到指定的分片上。
字符串Hash算法其实就是将字段值为字符串的内容,转换成二进制数,然后通过固定Hash计算出应该落在哪个节点上。
我们会在定义分片策略时,一共有2个分片节点,例如定义第一个节点存储位运算结果0-511之间的数据,第二个节点处理512-1023之间的数据,此时每个分片节点都处理512个数组,是均匀分配的现象。
在0-511之间数组的结构中,会记录上第一个分片节点的ID0,512-1023之间的数组的结构中,记录第二个分片节点的ID1。
当字段值为word时,首先根据截取的子字符串长度(0:2)截取3位,截取后子字符串为wor,然后通过Hash运算得到一个二进制数,再通过固定Hash,将二进制数与1111111111进行位运算,得到一个十进制数5,拿着这个十进制数5在数组集合中查找,最后查找到5位于0-511之内,0-511属于分片1,此时这条数据就会路由到分片1中存储。
需求:目前有一张tb_strhash表,表中的id一列的值是字符串类型,我们按照这个字段进行字符串Hash算法分片。
分片依旧是2个,还是之前垂直分库分表时使用的两套双主双从集群。
#在分片1节点中创建表结构
[root@mysql-1 ~]# mysql -uroot -p123456 -P3306 -h 192.168.20.11
mysql> use db_2;
mysql> create table tb_strhash (name varchar(20),content varchar(100));
#在分片2节点中创建表结构
[root@mysql-1 ~]# mysql -uroot -p123456 -P3307 -h 192.168.20.11
mysql> use db_2;
mysql> create table tb_strhash (name varchar(20),content varchar(100));
[root@mysql-1 ~]# vim /data/mycat/conf/schema.xml <?xml version="1.0"?> <!DOCTYPE mycat:schema SYSTEM "schema.dtd"> <mycat:schema xmlns:mycat="http://io.mycat/"> <!--定义逻辑库 库名叫做db_shopping 该逻辑库关联dn1这个数据节点--> <schema name="db_2" checkSQLschema="false" sqlMaxLimit="100" dataNode="dn1"> <!-- 字符串hash解析算法 --> <table name="tb_strhash" dataNode="dn1,dn2" rule="sharding-by-stringhash" /> </schema> <!--定义数据节点 也就是分片 一个分片会关联一个数据主机组 然后对应真实的数据库名称--> <dataNode name="dn1" dataHost="mysqlcluster-1" database= "db_2" /> <dataNode name="dn2" dataHost="mysqlcluster-2" database= "db_2" /> <!--定义数据主机 在这个标签下定义具体的读写操作路由的数据库实例地址 schema、table划分如何指定的是该数据主机关联的数据节点 那么对应的库、表都会被存储在数据主机定义的数据库实例中--> <dataHost name="mysqlcluster-1" maxCon="1000" minCon="10" balance="1" writeType="0" dbType="mysql" dbDriver="native" switchType="1"> <heartbeat>select user()</heartbeat> <!--定义写操作路由的数据库实例--> <writeHost host="c1-1-master3306" url="192.168.20.11:3306" user="root" password="123456"> <!--定义读操作路由的数据库实例--> <readHost host="c1-1-slave3308" url="192.168.20.11:3308" user="root" password="123456" /> </writeHost> <!--备用的主库 也是提供写操作的数据库,当主库c1-1-master3306故障后 备用库开始提供写操作--> <writeHost host="c1-2-master3306" url="192.168.20.12:3306" user="root" password="123456"> <!--备用主库的从库 从始至终 只要备用主库不故障 会一直提供读服务--> <readHost host="c1-2-slave3308" url="192.168.20.12:3308" user="root" password="123456" /> </writeHost> </dataHost> <dataHost name="mysqlcluster-2" maxCon="1000" minCon="10" balance="1" writeType="0" dbType="mysql" dbDriver="native" switchType="1"> <heartbeat>select user()</heartbeat> <writeHost host="c2-1-master3307" url="192.168.20.11:3307" user="root" password="123456"> <readHost host="c2-1-slave3309" url="192.168.20.11:3309" user="root" password="123456" /> </writeHost> <!--备用主库db3 主库db1故障后 开始提供写操作--> <writeHost host="c2-2-master3307" url="192.168.20.12:3307" user="root" password="123456"> <!--备用主库的从库 从始至终 只要备用主库不故障 会一直提供读服务--> <readHost host="c2-2-slave3309" url="192.168.20.12:3309" user="root" password="123456" /> </writeHost> </dataHost> </mycat:schema>
[root@mysql-1 ~]# vim /data/mycat/conf/rule.xml <tableRule name="sharding-by-stringhash"> <rule> <columns>name</columns> <algorithm>sharding-by-stringhash</algorithm> </rule> </tableRule> <function name="sharding-by-stringhash" class="io.mycat.route.function.PartitionByString"> <!--每个分片所承载的数组数量--> <property name="partitionLength">512</property> <!--分片节点的数量--> <property name="partitionCount">2</property> <!--字符串的截取长度,0:2表示截取第0/1/2位,一共3位长度的子字符串--> <property name="hashSlice">0:2</property> </function>
[root@mysql-1 ~]# vim /data/mycat/conf/server.xml
<user name="root" defaultAccount="true">
<!--登录用户的密码-->
<property name="password">123456</property>
<!--该用户登录后可以显示那些Schema-->
<property name="schemas">db_2</property>
</user>
[root@mysql-1 ~]# mycat restart
Stopping Mycat-server...
Stopped Mycat-server.
Starting Mycat-server...
[root@mysql-1 ~]# mysql -uroot -p123456 -P8066 -h 192.168.20.11
mysql> use db_2;
mysql> insert into tb_strhash (name,content) values ('T1001', UUID());
mysql> insert into tb_strhash (name,content) values ('ROSE', UUID());
mysql> insert into tb_strhash (name,content) values ('JERRY', UUID());
mysql> insert into tb_strhash (name,content) values ('CRISTINA', UUID());
mysql> insert into tb_strhash (name,content) values ('TOMCAT', UUID());
可以看到一部分数据写入到了分片1中,一部分数据写入到了分片2中。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。