Java--Spring项目生成雪花算法数字(Twitter SnowFlake)

文章目录

        • 前言
        • 步骤
        • 查看结果

前言
  • 分布式系统常需要全局唯一的数字作为id,且该id要求有序,twitter的SnowFlake解决了这种需求,生成了符合条件的这种数字,本文将提供一个接口获取雪花算法数字。以下为代码。
步骤
  1. SnowFlakeUtils 雪花算法工具类

    @Slf4j
    public class SnowFlakeUtils {private static final RedisOperation REDIS_OPERATION = ApplicationContextHelper.getBean(RedisOperation.class);private static final String LOCAL_IP = getLocalIp();private static volatile SnowFlakeUtils instance;/*** 该任务开始时间,必须手动设置(差值的唯一性)* 建议在生产部署时选择某一日的0时0分0秒0毫秒的时间戳,方便计算*/private static final long START_TIME = 1588733692671L;/*** 各个位的位数,Timestamp为41L(无需定义)*/private static final long DATA_CENTER_ID_BITS = 5L;private static final long WORKER_ID_BITS = 1L;private static final long SEQUENCE_BITS = 16L;/*** 各位的最大值*/private static final long DATA_CENTER_ID_MAX = ~(-1 << DATA_CENTER_ID_BITS);private static final long WORKER_ID_MAX = ~(-1 << WORKER_ID_BITS);private static final long SEQUENCE_MAX = ~(-1 << SEQUENCE_BITS);/*** 各位应该向左移动位数*/private static final long TIMESTAMP_SHIFT = SEQUENCE_BITS + WORKER_ID_BITS + DATA_CENTER_ID_BITS;private static final long DATA_CENTER_ID_SHIFT = SEQUENCE_BITS + WORKER_ID_BITS;private static final long WORKER_ID_SHIFT = SEQUENCE_BITS;/*** 数据中心ID*/private final long dataCenterId;private static final String DATA_CENTER_ID = "DATACENTERID";/*** 工作线程ID*/private final long workerId;private static final String WORKER_ID = "WORKERID";/*** 序列号*/private long sequence = 0L;/*** 上次时间(保证不回退)*/private long lastTimestamp = -1L;/**** 是否在高并发下*/private boolean isClock = false;public static SnowFlakeUtils getInstance() {if (instance == null) {synchronized (SnowFlakeUtils.class) {if (instance == null) {int dataCenterId = 0;int workerId = 0;while (true) {// tryCatch保证即使redis等出现问题也可以保证当前线程阻塞,重启redis即可处理继续处理try {String replace = RedisKeyConstant.SNOW_FLAKE_KEY.replace(DATA_CENTER_ID, String.valueOf(dataCenterId)).replace(WORKER_ID, String.valueOf(workerId));if (REDIS_OPERATION.setnx(replace, LOCAL_IP, 1, TimeUnit.MINUTES)) {instance = new SnowFlakeUtils(dataCenterId, workerId);break;}// 进行重新set直至成功,目前只运用dataCenterIdif (dataCenterId++ == DATA_CENTER_ID_MAX) {log.error("SnowFlake is getting CacheLock, please checkDATACENTERID_MAX={}", DATA_CENTER_ID_MAX);dataCenterId = 0;}} catch (Exception e) {log.error("SnowFlakeUtils get CacheLock Error, errorMsg:", e);try {Thread.sleep(MagicNum.THOUSAND);} catch (InterruptedException ex) {log.error(ex.getMessage(), ex);}}}}}}return instance;}public SnowFlakeUtils(long dataCenterId, long workerId) {if (dataCenterId > DATA_CENTER_ID_MAX || dataCenterId < 0) {throw new IllegalArgumentException(String.format("data center id can't be greater than %d or less than 0", DATA_CENTER_ID_MAX));}if (workerId > WORKER_ID_MAX || workerId < 0) {throw new IllegalArgumentException(String.format("worker id can't be greater than %d or less than 0", WORKER_ID_MAX));}this.dataCenterId = dataCenterId;this.workerId = workerId;String key = RedisKeyConstant.SNOW_FLAKE_KEY.replace(DATA_CENTER_ID, String.valueOf(dataCenterId)).replace(WORKER_ID, String.valueOf(workerId));log.info("SnowFlakeUtils Cache Key={}", key);// 起线程保证workerId和dataCenter组合不重复Thread thread = new Thread(new Runnable() {@Overridepublic void run() {while (true) {try {log.debug("SnowFlakeUtils is keep geting CacheLock-{}", key);String localIp = REDIS_OPERATION.get(key);if (LOCAL_IP.equals(localIp)) {REDIS_OPERATION.setex(key, LOCAL_IP, 1, TimeUnit.MINUTES);} else if (!REDIS_OPERATION.setnx(key, LOCAL_IP, 1, TimeUnit.MINUTES)) {throw new ProcessException(CommonConstants.ENUM_PROCESSING_EXCEPTION,"SnowFlakeUtils losed CacheLock-" + key + "." +"CacheLockKeeperThread broken!" +"Reday to retrieve CacheLock and Single Instance!");}Thread.sleep(MagicNum.FIFTY * MagicNum.THOUSAND);} catch (Exception e) {// 发生异常 将单例清除 并退出循环结束子线程synchronized (SnowFlakeUtils.class) {instance = null;}log.error(e.getMessage(),e);break;}}}});thread.setName("SnowFlake-CacheLockKeeper-" + dataCenterId + "-" + workerId);thread.start();}public void setClock(boolean clock) {this.isClock = clock;}public synchronized long nextId() {long timestamp = this.getTime();if (timestamp < lastTimestamp) {long offset = lastTimestamp - timestamp;if (offset <= MagicNum.FIVE) {try {this.wait(offset << 1);timestamp = this.getTime();if (timestamp < lastTimestamp) {throw new RuntimeException(String.format("Clock moved backwards, Refusing to generate id for %d milliseconds", offset));}} catch (InterruptedException e) {log.error(e.getMessage(), e);}} else {throw new RuntimeException(String.format("Clock moved backwards, Refusing to generate id for %d milliseconds", offset));}}if (lastTimestamp == timestamp) {sequence = sequence + 1;if (sequence > SEQUENCE_MAX) {timestamp = tilNextMillis(timestamp);sequence = 0;}} else {sequence = 0;}lastTimestamp = timestamp;return ((timestamp - START_TIME) << TIMESTAMP_SHIFT) |(dataCenterId << DATA_CENTER_ID_SHIFT) |(workerId << WORKER_ID_SHIFT) |sequence;}/*** 该毫秒达到上限,等待到下1毫秒*/private long tilNextMillis(long timestamp) {while (getTime() <= timestamp) {log.debug("单毫秒主键生成达到上限");}return this.getTime();}private long getTime() {if (isClock) {return SystemClock.currentTimeMillis();} else {return System.currentTimeMillis();}}private static String getLocalIp() {String ip = "";try {InetAddress addr = InetAddress.getLocalHost();ip += addr.getHostAddress();} catch (Exception e) {ip += "127.0.0.1";}ip += "_" + System.currentTimeMillis() + "_" + Math.random();log.info("SnowFlakeUtils Cache Value={}", ip);return ip;}
    }
    
  2. SystemClock

    /*** 由于高并发,在同一毫秒中会多次获取currentTimeMillis,而每次使用System.currentTimeMillis都会占用CPU(native方法).* 于是自定义类(single)来获取currentTimeMillis,实现方法是在此类中定义时间并设置一个周期任务(定时线程)1毫秒更新类中的时间*/
    public final class SystemClock {private static final SystemClock INSTANCE = new SystemClock(1);public static SystemClock getInstance() {return INSTANCE;}/*** 更新时间的时间间隔,默认为1毫秒*/private final long period;/*** 当前时间*/private final AtomicLong now;private SystemClock(long period) {this.period = period;this.now = new AtomicLong(System.currentTimeMillis());scheduleClockUpdate();}/*** 定时任务(设置为守护线程,1毫秒后开始更新)* scheduleAtFixedRate: 每次开始间隔为1毫秒* scheduleWithFixedDelay: 每次结束与开始为1毫秒*/private void scheduleClockUpdate() {ScheduledExecutorService executorService = Executors.newSingleThreadScheduledExecutor(new ThreadFactory() {@Overridepublic Thread newThread(Runnable r) {Thread thread = new Thread(r, "System Clock");thread.setDaemon(true);return thread;}});executorService.scheduleAtFixedRate(new Runnable() {@Overridepublic void run() {now.set(System.currentTimeMillis());}}, period, period, TimeUnit.MILLISECONDS);}public static long currentTimeMillis() {return getInstance().now.get();}
    }
    
  3. ApplicationContextHelper

    @Slf4j
    @Component
    public class ApplicationContextHelper implements ApplicationContextAware {
    /**
    * Spring上下文
    */
    private static ApplicationContext applicationContext;/*** @return ApplicationContext*/public static ApplicationContext getApplicationContext() {return applicationContext;}/*** 获取ApplicationContextAware**/@Overridepublic void setApplicationContext(ApplicationContext applicationContext) {ApplicationContextHelper.applicationContext = applicationContext;}/*** 根据Class获取对应实例**/public static <T> T getBean(Class<T> clz) {return applicationContext.getBean(clz);}/*** 根据beanName获取对应实例*/public static <T> T getBean(String name, Class<T> requiredType) {return applicationContext.getBean(name, requiredType);}public static Object getBean(String name) {return applicationContext.getBean(name);}
    }
    
  4. RedisOperation获取:RedisOperation,Redis操作工具类

  5. 在Controller里编写接口

    @RestController
    @RequestMapping("/part/util")
    public class UtilController {@ApiOperation("获取雪花数字")@GetMapping("/getSnowFlakeNo")public Result getSnowFlakeNo() {return Result.ok().data(String.valueOf(SnowFlakeUtils.getInstance().nextId()));}
    }
    
查看结果
  • 启动项目,有postman访问接口,查看结果如下,返回结果中data的值即为雪花算法数字。
    在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/612626.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux Ubuntu 20.04.6 Intel WiFi6 Ax411 1690i Ax1690i Killer 解决无线网卡识别不出来问题

项目场景&#xff1a; 网卡型号&#xff1a;英特尔 Killer™ Wi-Fi 6E AX1690 i/s ubuntu 版本 uname -a Linux kuanli 5.15.0-91-generic #101~20.04.1-Ubuntu SMP Thu Nov 16 14:22:28 UTC 2023 x86_64 x86_64 x86_64 GNU/Linux cat /proc/version Linux version 5.15.0-9…

windows11运行bat系统找不到指定的路径。

当运行.bat文件时出现“系统找不到指定的路径”错误通常是因为指定的路径不存在或者路径中包含了错误的字符。有几种方法可以解决这个问题&#xff1a; 检查路径是否存在&#xff1a;确保.bat文件中指定的路径是正确的&#xff0c;并且文件或文件夹确实存在。可以使用文件资源管…

docker 部署项目的操作文档,安装nginx

目录 1 部署环境检查2 相关知识点2.1 docker默认镜像存放地址2.2 docker 的镜像都是tar 包&#xff1f;2.3 Docker-compose 是直接使用镜像创建容器&#xff1f;2.4 Docker Compose down 就是将容器删除&#xff1f;2.5 删除&#xff0c;会删除挂载嘛2.6 DockerFile 和 docker …

实现本地存储函数useLocalStorage

我们经常需要使用 LocalStorage API&#xff0c;一个好用的可组合函数封装将帮助我们更好地使用它&#xff0c;让我们开始吧 &#x1f447;: <script setup langts>import { ref, watch } from "vue"/*** Implement the composable function* Make sure the f…

前端随机验证码安全验证sdk

前端随机验证码安全验证sdk 前言介绍一、效果展示二、使用步骤1.引入库2.参数说明3.方法与事件说明4.如何通过API获取当前用户的验证状态 ​ 前端必备工具推荐网站(免费图床、API和ChatAI等实用工具): http://luckycola.com.cn/ 前言 验证码&#xff1a;是一种校验区分用户是…

vue项目打包优化:缩小体积productionSourceMap设置,使用cdn加速

文章目录 一、vue项目打包体积大优化之productionSourceMap设置1、productionSourceMap 的作用2、禁用 productionSourceMap3、关闭 productionSourceMap4、配置 productionSourceMap 二、vue-cli打包之性能优化-使用cdn加速1、CDN加速是什么2、CDN加速具有以下优点&#xff1a…

Visual Studio 新特性:对 include 指令进行智能诊断

今天&#xff0c;我们很高兴地宣布新功能&#xff1a;#include 语言智能诊断。 此功能自 Visual Studio 2022 v17.9 预览版2 中可用。通过此新功能&#xff0c;您可以获取到有关每个 include 的引用和生成时间的详细信息&#xff0c;从而更好地了解 #include 指令的行为。 &g…

git仓库操作之一:git仓库修改名称

1 先修改“Project name"方法如下&#xff1a; 2 再修改“下载地址和下载后的项目名称”方法如下&#xff1a; 这样就修改完成了。

PhpPythonC++圆类的实现(OOP)

哎......被投诉了 &#x1f62d;&#x1f62d;&#x1f62d;&#x1f62d;&#x1f62d; 其实也不是小编不更&#xff0c;这不是期末了吗&#xff08;zhaojiekou~~&#xff09;&#xff0c;而且最近学的信息收集和ctf感觉好像没找到啥能更的&#xff08;不过最经还是在考虑更一…

JAVA销售数据决策管理系统源码

JAVA销售数据决策管理系统源码 基于BS&#xff08;Extjs Strus2springhibernate Mysql&#xff09;的销售数据的决策支持 主要的功能有 系统功能具体内容包括基础资料、进货管理、出货管理、库存管理、决策分析、系统管理。

活水计划丨改善老年营养,促进老年健康

在中国乡村发展基金会、腾讯公益的支持下&#xff0c;1月10日上午辉县义工联合共济医院&#xff0c;在我市易地搬迁佳怡社区开展“乐伴银龄 社区共建”项目——“老年常见病健康知识讲座”活动。 通过制作宣传横幅、发放科普手册等方式&#xff0c;为协同推进健康中国和积极应对…

《AI基本原理和python实现》栏目介绍

一、说明 栏目《AI基本原理和python实现》的设计目的是为了实现相关算法的python编程。因为用python实现AI需对相关的python库进行全方位了解&#xff0c;本栏目基本包含了【机器学习】相关的经典算法&#xff0c;除此之外还包括了数据分析、时间序列等一些概念和相关python代码…

【SpringCloud Alibaba】Nacos Config配置管理与Gateway 网关

目录 一、Config 远程配置 1.1 config 介绍 1.2 bootstrap.yml 配置文件 二、Gateway 网关 2.1 gateway 介绍 2.2 gateway 使用 2.2.1 方式一 2.2.2 方式二&#xff08;动态路由&#xff09; 一、Config 远程配置 1.1 config 介绍 微服务意味着要将单体应用中的业务拆分…

【Linux】Ubuntu 解压 zip、z01、z02等压缩文件的方法,Linux如何解压分卷压缩的

zip分卷压缩&#xff0c;在windows上压缩来的&#xff0c;如何解压这种文件&#xff1a; -rw-rw-r-- 1 20401094656 Dec 10 20:06 FFHQ.z01 -rw-rw-r-- 1 20401094656 Dec 10 20:10 FFHQ.z02 -rw-rw-r-- 1 20401094656 Dec 10 23:22 FFHQ.z03 -rw-rw-r-- 1 20401094656 Dec 10…

PyQt QTextEdit 详解

PyQt QTextEdit 详解 QTextEdit 是 PyQt 中用于编辑和显示多行文本的组件。它允许用户输入、编辑和格式化文本&#xff0c;并支持丰富的文本编辑功能。以下是关于 QTextEdit 的一些详细解释和示例&#xff1a; 创建 QTextEdit 对象&#xff1a; 要创建一个 QTextEdit 对象&a…

5.MapReduce之Combiner-预聚合

目录 概述本地预计算 Combiner 意义实践前提代码日志观察 结束 概述 在 MR、Spark、Flink 中&#xff0c;常用的减少网络传输的手段。 通常在 Reducer 端合并&#xff0c;shuffle 的数据量比在 Mapper 端要大&#xff0c;根据业务情况及数据量极大时&#xff0c;将大幅度降低效…

[足式机器人]Part3 机构运动学与动力学分析与建模 Ch00-3(1) 刚体的位形 Configuration of Rigid Body

本文仅供学习使用&#xff0c;总结很多本现有讲述运动学或动力学书籍后的总结&#xff0c;从矢量的角度进行分析&#xff0c;方法比较传统&#xff0c;但更易理解&#xff0c;并且现有的看似抽象方法&#xff0c;两者本质上并无不同。 2024年底本人学位论文发表后方可摘抄 若有…

pyqt treeWidget树生成

生成treeWidget树与获取treeWidget树节点的数据 # encodingUTF-8 import sys from PyQt5.QtCore import Qt from PyQt5.QtWidgets import QApplication, QTreeWidgetItem, QLineEdit, QSpinBox, QComboBox from PyQt5.QtWidgets import QWidget from release_test import Ui_F…

K8S容器编排高级应用

K8S容器编排高级应用 1.Pod控制器 pod控制器帮助我们自动管理pod&#xff0c;并满足期望的pod数量。pod控制器通过label标签来管理pod。在资源文件中通过selector来配置选择器&#xff0c;通过kind来配置控制器。一般我们的应用在生产环境用k8s一定要用pod控制器管理pod而不是…

Intellij-idea 如何编译maven工程

在 IntelliJ IDEA 中编译 Maven 工程是一个相对直接的过程。以下是基本步骤&#xff1a; 1. 打开或导入 Maven 项目 如果您已经有一个现有的 Maven 项目&#xff0c;可以直接在 IntelliJ IDEA 中打开它。选择 File > Open&#xff0c;然后浏览到您的 Maven 项目文件夹&…