snowflake mysql_SnowFlake 生成全局唯一id

public classSnowFlakeUtil {private longworkerId;private longdatacenterId;private long sequence = 0L;private long twepoch = 1288834974657L; //Thu, 04 Nov 2010 01:42:54 GMT 标记时间 用来计算偏移量,距离当前时间不同,得到的数据的位数也不同

private long workerIdBits = 5L; //物理节点ID长度

private long datacenterIdBits = 5L; //数据中心ID长度

private long maxWorkerId = -1L ^ (-1L << workerIdBits); //最大支持机器节点数0~31,一共32个

private long maxDatacenterId = -1L ^ (-1L << datacenterIdBits); //最大支持数据中心节点数0~31,一共32个

private long sequenceBits = 12L; //序列号12位, 4095,同毫秒内生成不同id的最大个数

private long workerIdShift = sequenceBits; //机器节点左移12位

private long datacenterIdShift = sequenceBits + workerIdBits; //数据中心节点左移17位

private long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits; //时间毫秒数左移22位

private long sequenceMask = -1L ^ (-1L << sequenceBits); //用于和当前时间戳做比较,以获取最新时间

private long lastTimestamp = -1L;/*** 成员类,SnowFlakeUtil的实例对象的保存域

**/

private static classIdGenHolder {private static final SnowFlakeUtil instance = newSnowFlakeUtil();

}/*** 外部调用获取SnowFlakeUtil的实例对象,确保不可变

*

*@return

*/

public staticSnowFlakeUtil get() {returnIdGenHolder.instance;

}/*** 初始化构造,无参构造有参函数,默认节点都是0*/

publicSnowFlakeUtil() {this(0L, 0L);

}/*** 设置机器节点和数据中心节点数,都是 0-31

*

*@paramworkerId

*@paramdatacenterId*/

public SnowFlakeUtil(long workerId, longdatacenterId) {if (workerId > maxWorkerId || workerId < 0) {throw newIllegalArgumentException(

String.format("worker Id can't be greater than %d or less than 0", maxWorkerId));

}if (datacenterId > maxDatacenterId || datacenterId < 0) {throw newIllegalArgumentException(

String.format("datacenter Id can't be greater than %d or less than 0", maxDatacenterId));

}this.workerId =workerId;this.datacenterId =datacenterId;

}/*** 线程安全的id生成方法

*

*@return

*/@SuppressWarnings("all")public synchronized longnextId() {//获取当前毫秒数

long timestamp =timeGen();//如果服务器时间有问题(时钟后退) 报错。

if (timestamp

}//如果上次生成时间和当前时间相同,在同一毫秒内

if (lastTimestamp ==timestamp) {//sequence自增,因为sequence只有12bit,所以和sequenceMask相与一下,去掉高位

sequence = (sequence + 1) &sequenceMask;//判断是否溢出,也就是每毫秒内超过4095,当为4096时,与sequenceMask相与,sequence就等于0

if (sequence == 0) {//自旋等待到下一毫秒

timestamp =tilNextMillis(lastTimestamp);

}

}else{//如果和上次生成时间不同,重置sequence,就是下一毫秒开始,sequence计数重新从0开始累加,每个毫秒时间内,都是从0开始计数,最大4095

sequence = 0L;

}

lastTimestamp=timestamp;//最后按照规则拼出ID 64位//000000000000000000000000000000000000000000 00000 00000 000000000000//1位固定整数 time datacenterId workerId sequence

return ((timestamp - twepoch) << timestampLeftShift) | (datacenterId <

}/*** 比较当前时间和过去时间,防止时钟回退(机器问题),保证给的都是最新时间/最大时间

*

*@paramlastTimestamp

*@return

*/

protected long tilNextMillis(longlastTimestamp) {long timestamp =timeGen();while (timestamp <=lastTimestamp) {

timestamp=timeGen();

}returntimestamp;

}/*** 获取当前的时间戳(毫秒)

*

*@return

*/

protected longtimeGen() {returnSystem.currentTimeMillis();

}/*** 获取全局唯一id*/

public staticString getId() {

Long id=SnowFlakeUtil.get().nextId();returnid.toString();

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值