使用 Mycat 中间件搭建 MySQL 高可用实现分库分表及读写分离

发布于 2019-12-06

使用 Mycat 中间件搭建 MySQL 高可用实现分库分表及读写分离

Mycat 是一款基于阿里开源产品Cobar而研发的开源数据库分库分表中间件(基于Java语言开发),可以用来方便地搭建面向企业应用开发的大数据库集群,支持事务、ACID等特性,其核心是基于代理方案实现 MySQL 高可用,是目前比较流行的 MySQL 高可用的方案之一。

Mycat 官网地址:

http://www.mycat.io/

使用 Mycat 配置 MySQL 高可用,实现分库分表、读写分离及主从切换

架构图:

实例部署情况:

db1-M1,IP:192.168.0.103,Port:3306 (主1)
db1-M2,IP:192.168.0.104,Port:3306 (主2)
db1-M3,IP:192.168.0.105,Port:3306 (主3)
db1-S1,IP:192.168.0.106, Port:3306 (从1)

server.xml

配置两个帐号,一个拥有全部权限,另一个只有读权限,帐号及密码可以自由定义。

<user name="root" defaultAccount="true">
    <property name="password">123456</property>
    <property name="schemas">db</property>
</user>
		
<user name="user">
    <property name="password">user</property>
    <property name="schemas">db</property>
    <property name="readOnly">true</property>
</user>

schema.xml

分成三个片,其中第一个版配置读写分离,共4个shard。

<?xml version="1.0"?>
<!DOCTYPE mycat:schema SYSTEM "schema.dtd">
<mycat:schema xmlns:mycat="http://io.mycat/">

    <schema name="db" checkSQLschema="false" sqlMaxLimit="100">
        <table name="t_user" primaryKey="ID" dataNode="dn1,dn2,dn3" rule="mod-long" />
    </schema>
    
    <!-- 分为三个片 -->
    <dataNode name="dn1" dataHost="shard1" database="db1" />
    <dataNode name="dn2" dataHost="shard2" database="db2" />
    <dataNode name="dn3" dataHost="shard3" database="db3" />

    <dataHost name="shard1" maxCon="1000" minCon="10" balance="1" writeType="0" dbType="mysql" dbDriver="native" switchType="1" slaveThreshold="100">
        <heartbeat>select user()</heartbeat>
        <writeHost host="hostM1" url="192.168.0.103:3306" user="db1" password="123456">
            <!-- 配置读写分离 -->
            <readHost host="hostS1" url="192.168.0.106:3306" user="db1" password="123456" />
        </writeHost>
    </dataHost>
	
    <dataHost name="shard2" maxCon="1000" minCon="10" balance="0" writeType="0" dbType="mysql" dbDriver="native" switchType="1" slaveThreshold="100">
        <heartbeat>select user()</heartbeat>
        <writeHost host="hostM2" url="192.168.0.104:3306" user="db2" password="123456"/>
    </dataHost>
	
    <dataHost name="shard3" maxCon="1000" minCon="10" balance="0" writeType="0" dbType="mysql" dbDriver="native" switchType="1" slaveThreshold="100">
        <heartbeat>select user()</heartbeat>
        <writeHost host="hostM3" url="192.168.0.105:3306" user="db3" password="123456"/>
    </dataHost>
</mycat:schema>
提示

注:若要开启读写分离,dataHost节点中的balance参数设置为1。

  • 1. balance=”0″, 不开启读写分离机制,所有读操作都发送到当前可用的writeHost上。
  • 2. balance=”1″,全部的readHost与stand by writeHost参与select语句的负载均衡
  • 3. balance=”2″,所有读操作都随机的在writeHost、readhost上分发。
  • 4. balance=”3″,所有读请求随机的分发到wiriterHost对应的readhost执行,writerHost不负担读压力

rule.xml

主要配置项(本例未改动该配置文件):

<tableRule name="mod-long">
    <rule>
        <columns>id</columns> <!-- 分片列,根据id分片 -->
        <algorithm>mod-long</algorithm>
    </rule>
</tableRule>

<function name="mod-long" class="io.mycat.route.function.PartitionByMod">
    <!-- how many data nodes -->
    <property name="count">3</property> <!-- count值与分片个数相同,本例中分了3个片 -->
</function>

测试

使用 MySQL 客户端连接 MyCat 数据库,MyCat安装在本机,ip 是192.168.0.102,像正常连接 MySQL 那样,使用 root/123456,192.168.0.102 连接,其中端口号为 8066(可在 server.xml 中配置)。

如果配置没有问题的话,此处可以正常连接。

在 M1、M2、M3、S1 这四台数据库服务器上分别创建数据库 db1、db2、db3、db1(M1和S1 同属一个分片,所以库名一致),然后在各自库下创建 user 表(S1 下也需要创建 user 表),表结构如下:

create table user (
  id int(11) NOT NULL AUTO_INCREMENT,
  name varchar(8) DEFAULT NULL,
  PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8

创建成功后,会发现 Mycat 下也生成了虚拟的 user 表。

操作 Mycat 的 user 表,插入几条记录:

insert into user values (0, '用户0');
insert into user values (1, '用户1');
insert into user values (2, '用户2');
insert into user values (3, '用户3');
insert into user values (4, '用户4');
insert into user values (5, '用户5');
insert into user values (6, '用户6');
insert into user values (7, '用户7');
insert into user values (8, '用户8');
insert into user values (9, '用户9');

查看 M1、M2 和 M3 这三个分片各自 user 表下的数据,发现 id 为 0, 3, 6, 9 的数据落入了分片一,id 为 1, 4, 7 的数据落入了分片二,id 为 2, 5, 8  的数据落入了分片三。因为我们配置的是按 id 取余的算法来分库和分表的,所以可以确认我们配置的分库分表已生效。

如果配置了主从复制的话,会发现分片一 M1 和 S1 中的 user 表里的数据是一模一样的。关于MySQL 的主从复制,需要单独配置一下,可以参考 MySQL主从复制原理及配置教程 这篇文章。

接下来,我们该如何确定读写分离是否生效了呢?假设现在主库 M1 和从库 S1 中各有 10 条数据,数据当然是一模一样的,我们手动删除从库 S1 中的 1 条数据,然后通过 Mycat 查询分片一中的数据,会发现查询出来的数据总共有 9 条,这样,就可以确认读写分离生效了。

结束语

使用 Mycat 可以轻松实现 MySQL 的分库分表及读写分离,Mycat 因为是单节点的,自身也需要实现高可用,可以通过 keepalived 技术来实现。我们也可以官网或者 Mycat权威指南 来获取更多的配置信息。另外,Mycat 中间件只是用来配置 MySQL 高可用的方案之一,后续会介绍其它的 MySQL 高可用配置方案。

喜欢 0
奋楫笃行,臻于至善!

相关文章

通用架构模式和通用架构服务

架构模式是在给定上下文的软件架构中,针对常发生问题的一种通用、复用的解决方案。架构模式类似于软件设计模式,但是范畴更广。一个好的软件产品往往需要有良好的架构思想和架构服务来支撑整个软件的生命周期,本文...
阅读全文

Java 的可重入锁和不可重入锁

可重入锁又名递归锁,是指在同一个线程在外层方法获取锁的时候,再进入该线程的内层方法会自动获取锁(前提锁对象得是同一个对象或者class),不会因为之前已经获取过还没释放而阻塞。Java中Reentra...
阅读全文

Redis 的两种持久化方式及使用场景分析

Redis是内存数据库,数据都是存储在内存中,为了避免进程退出导致数据的永久丢失,需要定期将Redis中的数据以某种形式(数据或命令)从内存保存到硬盘。当下次Redis重启时,利用持久化文件实现数据恢...
阅读全文

redis 高可用主从,哨兵,集群解决方案

Redis因为其高性能和易用性在我们后端的服务中发挥了巨大的作用,并且很多重要功能的实现都会依赖redis。除了常用的缓存,还有队列,发布订阅等重要用处。所以redis的服务高可用就显得尤为关键。这里...
阅读全文

Redis 缓存穿透、缓存击穿、缓存雪崩的区别及解决方案

Redis缓存的使用,极大的提升了应用程序的性能和效率,特别是数据查询方面。但同时,它也带来了一些问题。其中,最要害的问题,就是数据的一致性问题,从严格意义上讲,这个问题无解。如果对数据的一致性要求很...
阅读全文

HashMap 在 JDK1.8 之前和之后的区别

哈希表(hash table),也叫散列表,是一种非常重要的数据结构,应用场景及其丰富,许多缓存技术(比如memcached)的核心其实就是在内存中维护一张大的哈希表,本文会对java集合框架中Has...
阅读全文

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注