Mycat(4)——全局序列号

本文详细介绍了Mycat全局序列号的四种实现方式,包括本地文件、数据库、本地时间戳以及分布式ZKID生成器。数据库方式中,通过创建序列表并结合自定义函数实现序列号的获取和更新,确保高并发下的效率和一致性。同时,提供了相关配置文件和数据库脚本示例,便于实际操作。
摘要由CSDN通过智能技术生成

一、全局序列号方案

(一)本地文件方式

Mycat 将 sequence 配置到文件中,当使用到 sequence 中的配置后,Mycat 会更新classpath 中的 sequence_conf.properties 文件中sequence 当前的值。
优点:本地加载,读取速度较快
缺点:抗风险能力差,Mycat 所在主机宕机后,无法读取本地文件

(二)数据库方式

利用数据库一个表 来进行计数累加。但是并不是每次生成序列都读写数据库,这样效率太低。Mycat 会预加载一部分号段到 Mycat 的内存中,这样大部分读写序列都是在内存中完成的。如果内存中的号段用完了 Mycat 会再向数据库要一次。Mycat 重启后会向数据库申请新的号段,原有号段会弃用。

(三)本地时间戳方式

全局序列ID= 64 位二进制 (42(毫秒)+5(机器 ID)+5(业务编码)+12(重复累加) 换算成十进制为 18 位数的long 类型,每毫秒可以并发 12 位二进制的累加。
优点:配置简单
缺点:18 位 ID 过长

(四)其他方式

1. 分布式zk ID生成器
<property name="sequnceHandlerType">3</property>

Zk 的连接信息统一在 myid.properties 的 zkURL 属性中配置。
基于 ZK 与本地配置的分布式 ID 生成器(可以通过 ZK 获取集群(机房)唯一 InstanceID,也可以通过配置文件配置 InstanceID).

2. zk递增方式
<property name="sequnceHandlerType">4</property>

Zk 的连接信息统一在 myid.properties 的 zkURL 属性中配置。

  • 配置文件:sequence_conf.properties
  • 只要配置好 ZK 地址和表名的如下属性
  • TABLE.MINID 某线程当前区间内最小值
  • TABLE.MAXID 某线程当前区间内最大值
  • TABLE.CURID 某线程当前区间内当前值
  • 文件配置的 MAXID 以及 MINID 决定每次取得区间,这个对于每个线程或者进程都有效
  • 文件中的这三个属性配置只对第一个进程的第一个线程有效,其他线程和进程会动态读取 ZK
3. 自主生成全局序列

可以利用 redis 的单线程原子性 incr 来生成序列,自主生成需要单独在工程中用 java 代码实现。

4. MyCAT 自增长主键

需要mysql 节点上数据表,定义 auto_increment

参考《Mycat权威指南》

二、数据库方式实现全局序列号

(一)原理

在数据库中建立一张表,存放 sequence 名称(name),sequence 当前值(current_value),步长(increment int 类型,每次读取多少个 sequence,假设为 K)等信息;

(二)Sequence 获取步骤

  1. 当初次使用该 sequence 时,根据传入的 sequence 名称,从数据库这张表中读取 current_value,和increment 到 MyCat 中,并将数据库中的 current_value 设置为原 current_value 值+increment 值。
  2. MyCat 将读取到 current_value+increment 作为本次要使用的 sequence 值,下次使用时,自动加 1,当使用 increment 次后,执行步骤 1相同的操作。
  3. MyCat 负责维护这张表,用到哪些 sequence,只需要在这张表中插入一条记录即可。若某次读取的sequence 没有用完,系统就停掉了,则这次读取的 sequence 剩余值不会再使用。

(三)文件配置

server.xml:

<system><property name="sequnceHandlerType">1</property></system>

注:sequnceHandlerType 需要配置为 1,表示使用数据库方式生成 sequence。

sequence_db_conf.properties:

ORDERS=dn1

表示ORDERS这个序列在dn1这个节点上

修改/etc/my.cnf:

 log_bin_trust_function_creators=1

配置在 mysql 数据库中可以执行函数

(四)数据库配置

在dn1中,建表:


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;

INSERT INTO MYCAT_SEQUENCE(NAME,current_value,increment) VALUES ('ORDERS', 400000, 100);

创建函数:

- 获取当前 sequence 的值 (返回当前值,增量)
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 ;

- 设置 sequence 值
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 ;


- 获取下一个 sequence 值
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 ;

重启mycat后,登录执行:

insert into orders (id,order_type,customer_id,amount) values (next value for MYCATSEQ_ORDERS,101,100,101400);

insert into orders (id,order_type,customer_id,amount) values (next value for MYCATSEQ_ORDERS,101,101,101500);

select * from orders;

全局序列号生效。

参考《Mycat权威指南》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值