Java利用时间戳生成三种唯一主键

/**
 * @Author: lgz
 * @Date: 2020/5/24 23:44
 */
public enum IdUtil {
    //有年月日的时间id
    DATE_TIME_ID {
        public long generateId() {
            return Long.parseLong(DateTimeFormatter.ofPattern("yyyyMMddHHmmssSSS")
                    .format(LocalDateTime.ofInstant(Instant.ofEpochMilli(super.generateId()), ZoneId.systemDefault())));
        }
    },
    //直接返回时间戳
    TIMESTAMP_ID {
        public long generateId() {
            return super.generateId();
        }
    },
    //时间戳左移10
    SHIFT_ID {
        public long generateId() {
            return super.generateId() << 10;
        }
    };


    //上一次的时间戳
    //在集群环境中,可以使用redis存储上一次时间
    private long lastTimestamp = -1L;

    //在集群环境中,可以使用分布式锁
    private Lock lock = new ReentrantLock();

    private long nowTimestamp;

    public long generateId() {
        lock.lock();
        nowTimestamp = 0;
        try {
            nowTimestamp = Instant.now().toEpochMilli();
            if (lastTimestamp == nowTimestamp) {
                lastTimestamp = waitNextMilli();
            } else if (lastTimestamp > nowTimestamp) {
                throw new RuntimeException("time error when generated the id.");
            } else {
                lastTimestamp = nowTimestamp;
            }
        } finally {
            lock.unlock();
        }
        if (nowTimestamp == 0) {
            throw new RuntimeException("unknown error when generated the id.");
        }

        return nowTimestamp;
    }

    private long waitNextMilli() {
        //判断是否进入下一毫秒
        //如果并发量特别高,会给cpu带来额外负荷
        do {
            nowTimestamp = Instant.now().toEpochMilli();
        } while (nowTimestamp == lastTimestamp);
        return nowTimestamp;

        //直接睡一毫秒
//        try {
//            TimeUnit.MILLISECONDS.sleep(1);
//        } catch (InterruptedException e) {
//            e.printStackTrace();
//        }
//        nowTimestamp = Instant.now().toEpochMilli();
//        return nowTimestamp;
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值