简介
无论分多少个节点,id不可能重复,这就是全局序列
三种方式
本地文件
在Mycat建一个文件,专门存second序列,假设初始值为1,每当有insert语句执行时,使其+1,无论分到哪个数据节点都不会重复。
Mycat部署的服务器,一旦宕机,这个文件就无法读取,即使有备机,也无法使用,抗风险能力较差,不推荐。
数据库方式
利用数据库一个表来进行计数累加,但是并不是每次生成序列都读写数据库,这样效率太低,Mycat会预加载一部分号段到Mycat的内存中,这样大部分读写序列都是在内存中完成的。如果内存中的号段用完了 Mycat会再向数据库要一次。
如果Mycat崩溃了 ,当它重启之后,会向数据库申请新的号段,原有号段会弃用,但是不会因此出现主键重复,推荐使用这种方式!
配置
-
首先在主机创建表、脚本
# 建表 CREATE TABLE MYCAT_SEQUENCE (NAME VARCHAR(50) NOT NULL,current_value INT NOT NULL,increment INT NOT NULL DEFAULT 100, PRIMARY KEY(NAME)) ENGINE=INNODB; # Mycat提供的3个函数 DELIMITER $$ CREATE FUNCTION mycat_seq_currval(seq_name VARCHAR(50)) RETURNS VARCHAR(64) DETERMINISTIC BEGIN DECLARE retval VARCHAR(64); SET retval="-999999999,null"; SELECT CONCAT(CAST(current_value AS CHAR),",",CAST(increment AS CHAR)) INTO retval FROM MYCAT_SEQUENCE WHERE NAME = seq_name; RETURN retval; END $$ DELIMITER; DELIMITER $$ CREATE FUNCTION mycat_seq_setval(seq_name VARCHAR(50),VALUE INTEGER) RETURNS VARCHAR(64) DETERMINISTIC BEGIN UPDATE MYCAT_SEQUENCE SET current_value = VALUE WHERE NAME = seq_name; RETURN mycat_seq_currval(seq_name); END $$ DELIMITER; DELIMITER $$ CREATE FUNCTION mycat_seq_nextval(seq_name VARCHAR(50)) RETURNS VARCHAR(64) DETERMINISTIC BEGIN UPDATE MYCAT_SEQUENCE SET current_value = current_value + increment WHERE NAME = seq_name; RETURN mycat_seq_currval(seq_name); END $$ DELIMITER; # 插入数据 INSERT INTO MYCAT_SEQUENCE(NAME,current_value,increment) VALUES ('ORDERS', 400000, 100);
-
修改
sequence_db_conf.properties
意思是 ORDERS这个序列在dn1这个节点上,具体dn1节点是哪台主机,查看
schema.xml
-
修改
server.xml
,默认是2,修改为1 -
向Mycat中插入数据
insert into `orders`(id,amount,customer_id,order_type) values(next value for MYCATSEQ_ORDERS,1000,101,102);
-
结果,插入成功
-
模拟Mycat宕机,重启后,重新插入数据,跳过之前号段的,步长+100
时间戳方式
一旦数据经过Mycat服务器,取Mycat主机的时间戳作为全局序列,并且时间戳会不断变大,还不会重复。就算是主机挂了,从机上线,只要保证主机从机的时间一致,那就可以继续使用,比较好的一点就是,不需要配置文件,直接取系统时间,实现方式较为简单。
但是它有一个很不好的缺点,它的字段太长了,有18位,比如说要在订单表这种数据量较大的表中作为id使用,就有点浪费空间了,所以这种方式也不推荐。
自主生成
如果要使用自主生成的全局序列,那么Java程序中也需要写对应代码。
- 可以根据业务逻辑组合
- 可以利用redis的单线程原子性incr来生成