生成12位短id,自增且不连续,永不重复,不依赖数据库

这篇具有很好参考价值的文章主要介绍了生成12位短id,自增且不连续,永不重复,不依赖数据库。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

基本思路:

设计模式:单例模式

是否加锁:是 synchronized

获取最后一次生成的时间戳值T0

限定初始时间为2023-08-01 00:00:00,获取当前时间时间戳T1,T1与初始时间的毫秒差值T2,转为16进制,转为字符串为r1,获取该字符串的长度L1

获取L2 (length - L1) ,获取L2位数字的16进制自增数值范围,取最大值max

现数据库批量导入数据速度为 n条/ms

平均步长为max/n,(0~平均步长)的平均数为max/n/2,假设使用平均步长最为随机步长范围,最终的值与max相差较远,大约后一半的数字没有被使用

将平均步长*2-平均步长*容错因子(0.1)的值作为我们随机步长的范围  容错因子:减小溢出概率

随机步长step = max/n*2 - max/n*0.1

获取T1

如果T1 == T0,序列值seqNum = seqNum + step (转为16进制),若seqNum > max,该线程暂停1毫秒后刷新r1

如果T1 > T0,序列值seqNum = 0 + step

设置T0

代码实现如下:

/**
 * 生成短id
 * @author mayu
 */
public class ShortIdWorker {

    /**
     * 初始时间限定为2023-08-01 00:00:00
     */
    private final static long START_STAMP = 1690819200000L;

    /**
     * 容错因子
     */
    private final static int FAULT_TOLERANCE_FACTOR = 10;

    /**
     * 默认长度
     */
    private final static int DEFAULT_ID_LENGTH = 12;

    /**
     * 数据库每毫秒可保存的数据,结合列的数量取值,建议实测后更改
     */
    private final static int DEFAULT_TRANSFER_SPEED_PER_MILLISECOND = 50;

    private final int length;

    private final int transferSpeedPerMillisecond;

    /**
     * 上次运行时间
     */
    private long lastStamp = -1L;

    /**
     * 增长序列
     */
    private int seqNum;

    private static ShortIdWorker instance;

    /**
     * 单例模式
     */
    public static ShortIdWorker getInstance() {
        if (null == instance) {
            instance = new ShortIdWorker();
        }
        return instance;
    }

    public static ShortIdWorker newInstance(int length, int transferSpeedPerMillisecond) {
        return new ShortIdWorker(length, transferSpeedPerMillisecond);
    }

    /**
     * 默认使用12位id,数据库每毫秒新增数据为50条
     */
    private ShortIdWorker() {
        this(DEFAULT_ID_LENGTH, DEFAULT_TRANSFER_SPEED_PER_MILLISECOND);
    }

    private ShortIdWorker(int length, int transferSpeedPerMillisecond) {
        this.length = length;
        this.transferSpeedPerMillisecond = transferSpeedPerMillisecond;
    }

    /**
     * @return 生成后的id
     * <p>
     * 例:757b12c001d3
     * 共length位id,前x位为时间戳差值的16进制,后y位为不固定步长的自增序列
     */
    public synchronized String nextId() {
        long now = now();
        // 获取16进制时间戳前缀
        String stampPrefix = getStampStr(now);
        // 获取第二段增长序列的长度l2
        int l2 = this.length - stampPrefix.length();
        // 获取l2位16进制的最大值
        int max = IntStream.range(0, l2).map(i -> 16).reduce(1, (a, b) -> a * b) - 1;
        // 获取增长的平均步长averageStepLength
        int averageStepLength = max / this.transferSpeedPerMillisecond;
        // 取步长范围
        // averageStepLength的平均值是averageStepLength/2,累加的情况下会有后一半的空间浪费问题,故取值为averageStepLength*2,平均值为averageStepLength
        // 取随机数的结果不可控,上行中列举的只是近似值,为防止多次溢出影响程序执行时间,再减去容错因子,减小溢出概率(容错因子建议在本地系统实测后更改)
        int randomStepLengthMax = (averageStepLength << 1) - (averageStepLength / FAULT_TOLERANCE_FACTOR);
        // 在步长范围内获取随机步长
        int randomStepLength = new Random().nextInt(randomStepLengthMax) + 1;
        // 当上次运行时间小于当前时间或第一次运行时,增长序列赋值为随机步长,设置最后运行时间
        if (this.lastStamp < now || this.lastStamp == -1L) {
            this.seqNum = randomStepLength;
            this.lastStamp = now;
        // 当上次运行时间与当前运行时间处于同一毫秒时
        } else if (this.lastStamp == now) {
            // 增长序列以随机步长为步长递增
            this.seqNum += randomStepLength;
            // 当增长序列大于最大值时
            if (this.seqNum > max) {
                // 程序暂停一毫秒
                LockSupport.parkNanos(TimeUnit.MILLISECONDS.toNanos(1));
                // 重新获取前缀,增长序列重新开始
                this.seqNum = randomStepLength;
                Long newNow = now();
                this.lastStamp = newNow;
                stampPrefix = getStampStr(newNow);
            }
        } else {
            // 时钟回拨,报错
            throw new IllegalStateException("Clock moved backwards.  Reject to generate id");
        }
        // 将增长序列转为16进制与时间戳拼接
        return stampPrefix + String.format("%0" + l2 + "X", new BigInteger(String.valueOf(this.seqNum), 10));
    }


    private String hex10To16(String str) {
        return String.format("%X", new BigInteger(str, 10));
    }

    private long now() {
        return System.currentTimeMillis();
    }

    /**
     * 获取传入时间与开始时间的间隔毫秒数,将结果转为16进制
     * @param now 时间戳
     * @return
     */
    private String getStampStr(Long now) {
        return hex10To16(String.valueOf(now - START_STAMP));
    }

        8位16进制可使用到4201年-03-20 07:32:15,后续时间戳所占位数自动变为9位,id总长度不变,不用担心id用尽的问题。

        代码中关于时间赋值的代码请谨慎改动,顺序颠倒会产生bug。文章来源地址https://www.toymoban.com/news/detail-682609.html

到了这里,关于生成12位短id,自增且不连续,永不重复,不依赖数据库的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Linux命令:重复多次后台运行且不保存输出,查看命令对应的进程数量

    要在后台重复运行 Linux 命令并查看对应的进程数量,你可以使用循环结构和后台运行符号 `` 结合起来。以下是一个示例: ```bash for i in {1..3}; do     your_command /dev/null 21 done ``` 命令 `your_command /dev/null 21 ` 的含义如下: 1. `` 符号表示重定向输出。在这个命令中,`your_command` 的标

    2024年02月22日
    浏览(47)
  • 【数据库】自增ID有什么坏处?什么样的场景下不使用自增ID?

    在MySQL中,数据表的主键一般采用id字段自增的形式。使用自增ID给我们带来不少便捷,但也有不少坏处,在一些场景下是不推荐使用自增ID的。 自增ID是在设计表时将id字段的值设置为自增的形式,这样当插入一行数据时无需指定id会自动根据前一字段的ID值+1进行填充。在My

    2024年02月07日
    浏览(50)
  • postgresql设置id自增

    创建序列: 将序列与表的列关联: 可选地,你可以设置序列的起始值、递增步长和最大值: 完成上述步骤后,每次向表中插入新记录时,ID 列都会自动递增。可以使用以下命令查看当前序列的值: 这就是在 PostgreSQL 中将 ID 进行递增的基本方法。

    2024年02月03日
    浏览(62)
  • 表id自增的方法

    数据库主键id自增的方法,列举了几种如下 一、数据库自增 (部分数据库支持) 创建表的时候设置id自增即可,或者后期修改表id自增 二、序列号 (适合oracle) 三、mybatis获取自增 (通用) sql语句被执行的时候会自动带上主键字段和填充值去执行,语句中就不用写id字段了

    2024年02月08日
    浏览(34)
  • hive创建唯一标识列(自增id)

    目录 一、需求 二、方法 1.row_number() 2.UUID 3.row_sequence() 三、对比 在某一张 hive 表中需要有一列去唯一标识某一行,有些类似于MySQL中的自增ID 使用UDF函数row_sequence(),必须在Hive环境要有hive-contrib相关jar包 执行查询语句 row_number函数是对整个数据集做处理,自增序列在当次排序

    2024年02月14日
    浏览(40)
  • sqlite插入语句id自增列问题

    sqlite给主键id设置AUTOINCREMENT自增在插入数据的时候报错table has x columns but x-1 values were supplied 为什么自增列要显示不提供,sqlite需要提供自增列table ResTools has 7 columns but 6 values were supplied SQL Statement:insert into ResTools values(\\\'管理系统winform+Sqlite\\\',\\\'项目\\\\C#程序设计\\\\C#winform项目\\\\管理系

    2024年01月18日
    浏览(29)
  • Navicat Premium创建表设置id自增

    我们在使用Navicat Premium的时候创建表时左下角没有id自增选项的 那怎么才能实现id自增呢    nextval(\\\'tooksto\\\'::regclass)

    2024年02月15日
    浏览(38)
  • 使用hive sql 为hive增加或者创建自增列,自增id的五种方式

    *注意:此篇完全是废话,是错误演示文档 创建带有自增ID的Hive表的方法是使用Hive中的 SERDE (序列化和反序列化)和 ROW FORMAT 来为表添加自增ID。具体步骤如下: 在上面的代码中,我们首先使用 CREATE TABLE 语句创建一个名为 your_table 的表,并指定了表的列和数据类型。

    2023年04月18日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包