package com.azt.easysign.common.util; /** * @Author * @Date * @Version 1.0 * @Description: 雪花算法ID生成器 生成的是线程安全的递增的ID,分布式系统内不会重复 */ import java.lang.management.ManagementFactory; import java.net.InetAddress; import java.net.NetworkInterface; public class IdWorker { /** * 开始时间截,用程序开始使用的时间 2022-05-05 11:40 */ private final long twepoch = 1651722037880L; /** * 机器id所占的位数 */ private final long workerIdBits = 5L; /** * 数据标识id所占的位数 */ private final long datacenterIdBits = 5L; /** * 支持的最大机器id,结果是31 (这个移位算法可以很快的计算出几位二进制数所能表示的最大十进制数) */ private final long maxWorkerId = -1L ^ (-1L << workerIdBits); /** * 支持的最大数据标识id,结果是31 */ private final long maxDatacenterId = -1L ^ (-1L << datacenterIdBits); /** * 序列在id中占的位数 */ private final long sequenceBits = 12L; /** * 机器ID向左移12位 */ private final long workerIdShift = sequenceBits; /** * 数据标识id向左移17位(12+5) */ private final long datacenterIdShift = sequenceBits + workerIdBits; /** * 时间截向左移22位(5+5+12) */ private final long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits; /** * 生成序列的掩码,这里为4095 (0b111111111111=0xfff=4095) */ private final long sequenceMask = -1L ^ (-1L << sequenceBits); /** * 工作机器ID(0~31) */ private long workerId; /** * 数据中心ID(0~31) */ private long datacenterId; /** * 毫秒内序列(0~4095) */ private long sequence = 0L; /** * 上次生成ID的时间截 */ private long lastTimestamp = -1L; public IdWorker(){ this.datacenterId = getDatacenterId(maxDatacenterId); this.workerId = getMaxWorkerId(datacenterId, maxWorkerId); } /** * 构造函数 * * @param workerId 工作ID (0~31) * @param datacenterId 数据中心ID (0~31) */ private IdWorker(long workerId, long datacenterId) { if (workerId > maxWorkerId || workerId < 0) { throw new IllegalArgumentException(String.format("worker Id can't be greater than %d or less than 0", maxWorkerId)); } if (datacenterId > maxDatacenterId || datacenterId < 0) { throw new IllegalArgumentException(String.format("datacenter Id can't be greater than %d or less than 0", maxDatacenterId)); } this.workerId = workerId; this.datacenterId = datacenterId; } /** * 获得下一个ID (该方法是线程安全的) * * @return long */ private synchronized long nextId() { long timestamp = timeGen(); timestamp = generateId(timestamp); return ((timestamp - twepoch) << timestampLeftShift) // | (datacenterId << datacenterIdShift) // | (workerId << workerIdShift) // | sequence; } private long generateId(long timestamp) { //如果当前时间小于上一次ID生成的时间戳,说明系统时钟回退过这个时候应当抛出异常 if (timestamp < lastTimestamp) { throw new RuntimeException( String.format("Clock moved backwards. Refusing to generate id for %d milliseconds", lastTimestamp - timestamp)); } //如果是同一时间生成的,则进行毫秒内序列 if (lastTimestamp == timestamp) { sequence = (sequence + 1) & sequenceMask; //毫秒内序列溢出 if (sequence == 0) //阻塞到下一个毫秒,获得新的时间戳 { timestamp = tilNextMillis(lastTimestamp); } } else//时间戳改变,毫秒内序列重置 { sequence = 0L; } //上次生成ID的时间截 lastTimestamp = timestamp; return timestamp; } /** * 获得下一个ID (string) **/ private synchronized String generateNextId() { long timestamp = timeGen(); timestamp = generateId(timestamp); //移位并通过或运算拼到一起组成64位的ID return String.valueOf(((timestamp - twepoch) << timestampLeftShift) | (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence); } /** * 阻塞到下一个毫秒,直到获得新的时间戳 * * @param lastTimestamp 上次生成ID的时间截 * @return 当前时间戳 */ private long tilNextMillis(long lastTimestamp) { long timestamp = timeGen(); while (timestamp <= lastTimestamp) { timestamp = timeGen(); } return timestamp; } /** * 返回以毫秒为单位的当前时间 * * @return 当前时间(毫秒) */ private long timeGen() { return System.currentTimeMillis(); } /** * <p> * 获取 maxWorkerId * </p> */ protected static long getMaxWorkerId(long datacenterId, long maxWorkerId) { StringBuffer mpid = new StringBuffer(); mpid.append(datacenterId); String name = ManagementFactory.getRuntimeMXBean().getName(); if (!name.isEmpty()) { /* * GET jvmPid */ mpid.append(name.split("@")[0]); } /* * MAC + PID 的 hashcode 获取16个低位 */ return (mpid.toString().hashCode() & 0xffff) % (maxWorkerId + 1); } /** * <p> * 数据标识id部分 * </p> */ protected static long getDatacenterId(long maxDatacenterId) { long id = 0L; try { InetAddress ip = InetAddress.getLocalHost(); NetworkInterface network = NetworkInterface.getByInetAddress(ip); if (network == null) { id = 1L; } else { byte[] mac = network.getHardwareAddress(); id = ((0x000000FF & (long) mac[mac.length - 1]) | (0x0000FF00 & (((long) mac[mac.length - 2]) << 8))) >> 6; id = id % (maxDatacenterId + 1); } } catch (Exception e) { System.out.println(" getDatacenterId: " + e.getMessage()); } return id; } /** * 生成订单前缀,26个字母随机俩位 * @return */ public static String createPrefix(Integer num) { String randomPrefix = ""; for (int i = 0; i < num; i++) { char c = (char) (Math.random() * 26 + 'A'); randomPrefix += c; } return randomPrefix; } /** * 随机N个小写字母 * @return */ public static String lowerCreatePrefix(Integer num) { String randomPrefix = ""; for (int i = 0; i < num; i++) { char c = (char) (Math.random() * 26 + 'a'); randomPrefix += c; } return randomPrefix; } /** * @param: length 位数 * @description: TODO 生成N位随机数字 * @return: java.lang.Integer */ public static Integer randomNumber(int length){ return (new Double(Math.random() * (Math.pow(10F, length) - Math.pow(10F, length - 1)) + Math.pow(10F, length - 1))).intValue(); } //idWorker单例的,多例或分布式系统的话构造参数不能与其他实例相同,否则会产生安全问题 //private static IdWorker idWorker = new IdWorker(0L, 0L); private static IdWorker idWorker = new IdWorker(); /** * @Description: 公开方法,获取订单号 */ public static String getOrderNo(Integer num) { return createPrefix(num)+idWorker.nextId(); } /** * @Description: 公开方法,生成一个ID */ public static String getNextId() { return Long.toString(idWorker.nextId()); } public static void main(String[] args) { System.out.println("ID:"+IdWorker.getNextId()); } }
方法:
IdWorker.getNextId();