运用场景:
对于大型的互联网应用来说,数据库单表的记录行数可能达到千万级甚至是亿级,并且数据库面临着极高的并发访问。我这边有一个单表过亿的表数据,目前查询起来极为困难
解决方案:

  1. 制定分表策略
    在分表之前,首先需要选择适当的分表策略,使得数据能够较为均衡地分不到多张表中,并且不影响正常的查询!
    对于互联网企业来说,大部分数据都是与用户关联的,因此,用户id是最常用的分表字段。因为大部分查询都需要带上用户id,这样既不影响查询,又能够使数据较为均衡地
    分布到各个表中(当然,有的场景也可能会出现冷热数据分布不均衡的情况)
    在这里我就根据id进行分表
  2. 如何分表
    两种解决方案:垂直拆分、水平拆分
    垂直拆分:根据业务进行拆分,比如可以将一张表中的多个字段拆成两张表,一张是不经常更改的,一张是经常改的。
    水平拆分:即根据表来进行分割:比如user表可以拆分为user0,、user1、user2、user3、user4等
  3. 利用merge存储引擎来实现分表
    我觉得这种方法比较适合,那些没有事先考虑,而已经出现了得,数据查询慢的情况。这个时候如果要把已有的大数据量表分开比较痛苦,最痛苦的事就是改代码
    我们先建一个user_1表
CREATE TABLE `user_1` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(50) DEFAULT NULL,
  `age` int(1) NOT NULL DEFAULT '0',
  PRIMARY KEY (`id`)
) ENGINE=MyISAM DEFAULT CHARSET=utf8;

mysql 大表拆分小表 数据迁移 mysql分表实践_分区表


再创建一个user_2表

CREATE TABLE `user_2` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(50) DEFAULT NULL,
  `age` int(1) NOT NULL DEFAULT '0',
  PRIMARY KEY (`id`)
) ENGINE=MyISAM DEFAULT CHARSET=utf8;

mysql 大表拆分小表 数据迁移 mysql分表实践_分区表_02


创建一个user总表:

CREATE TABLE `user_all` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(50) DEFAULT NULL,
  `age` int(1) NOT NULL DEFAULT '0',
  KEY `id` (`id`)
) ENGINE=MRG_MyISAM DEFAULT CHARSET=utf8 INSERT_METHOD=LAST UNION=(`user_1`,`user_2`);

mysql 大表拆分小表 数据迁移 mysql分表实践_触发器_03

  • ENGINE = MERGE 和 ENGINE = MRG_MyISAM是一样的意思,都是代表使用的存储引擎是 Merge。
  • INSERT_METHOD,表示插入方式,取值可以是:0 和 1,0代表不允许插入,1代表可以插入;
  • FIRST插入到UNION中的第一个表,LAST插入到UNION中的最后一个表。

触发器

create table tb_ids(id int);
insert into tb_ids values(1);

然后在user_1和user_2表中分别建立一个触发器(tr_seq和tr_seq2),触发器的功能是 当在user_1或者user_2表中增加一条记录时,取出tb_ids中的id值,赋给user_1和user_2的id,然后将tb_ids的id值加1.
user_1的触发器:tr_seq

DELIMITER $$
   CREATE TRIGGER tr_seq
   BEFORE INSERT on user_1
   FOR EACH ROW BEGIN
      select id  into @testid from tb_ids limit 1;
      update tb_ids set id = @testid + 1;
   set new.id =  @testid;
   END$$
   DELIMITER;

user_2的触发器:tr_seq

DELIMITER $$
   CREATE TRIGGER tr_seq2
   BEFORE INSERT on user_2
   FOR EACH ROW BEGIN
      select id  into @testid from tb_ids limit 1;
      update tb_ids set id = @testid + 1;
   set new.id =  @testid;
   END$$
   DELIMITER;

此时已经分表成功
插入数据测试一下

insert into user_1(name,age) values('zhangsan','18');
insert into user_2(name,age) values('wangwu','20');
insert into user_all(name,age) values('wangwu','20');

user_2表

mysql 大表拆分小表 数据迁移 mysql分表实践_分区表_04


插入总表我们上面设置的是写在最后一张表里

mysql 大表拆分小表 数据迁移 mysql分表实践_分区表_05


user_all表数据

mysql 大表拆分小表 数据迁移 mysql分表实践_mysql_06


简单的水平分表完成