springBoot中雪花算术法

在 Spring Boot 中,雪花算法(Snowflake Algorithm)通常指的是 Twitter 开发的一种分布式唯一 ID 生成算法。它被广泛用于分布式系统中生成全局唯一的 ID,尤其是在高并发场景下。雪花算法生成的 ID 是一个 64 位的长整型数字,具有时间有序性和唯一性。

虽然 Spring Boot 本身没有直接内置雪花算法的实现,但你可以通过自定义代码或引入第三方库来实现它。下面我将解释雪花算法的原理,并提供一个在 Spring Boot 中实现的示例。

---

### 雪花算法原理
雪花算法生成的 64 位 ID 由以下部分组成:
1. **1 位符号位**:通常为 0,表示正数。
2. **41 位时间戳**:表示毫秒级时间戳,通常是当前时间与某个起始时间(epoch)的差值,可支持约 69 年的时间范围。
3. **10 位机器 ID**:表示机器或进程的标识,支持 1024 个节点。
4. **12 位序列号**:每毫秒内的自增序列号,支持每毫秒生成 4096 个 ID。

生成的 ID 结构如下:
```
0 | 41-bit timestamp | 10-bit worker ID | 12-bit sequence
```

优点:
- 高性能、高并发下仍能保证唯一性。
- ID 是时间有序的,便于排序和存储。
- 不依赖数据库等外部系统。

---

### 在 Spring Boot 中实现雪花算法
以下是一个简单的雪花算法实现示例,你可以将其集成到 Spring Boot 项目中。

#### 1. 创建雪花算法工具类
```java
public class SnowflakeIdGenerator {
    // 起始时间戳 (例如 2023-01-01 00:00:00)
    private static final long START_TIMESTAMP = 1672531200000L;

    // 各部分位数
    private static final long WORKER_ID_BITS = 10L; // 机器 ID 占 10 位
    private static final long SEQUENCE_BITS = 12L;  // 序列号占 12 位

    // 最大值
    private static final long MAX_WORKER_ID = ~(-1L << WORKER_ID_BITS); // 1023
    private static final long MAX_SEQUENCE = ~(-1L << SEQUENCE_BITS);   // 4095

    // 位移量
    private static final long WORKER_ID_SHIFT = SEQUENCE_BITS;
    private static final long TIMESTAMP_SHIFT = SEQUENCE_BITS + WORKER_ID_BITS;

    private long workerId;    // 机器 ID
    private long sequence = 0L; // 序列号
    private long lastTimestamp = -1L; // 上次生成 ID 的时间戳

    public SnowflakeIdGenerator(long workerId) {
        if (workerId > MAX_WORKER_ID || workerId < 0) {
            throw new IllegalArgumentException("Worker ID must be between 0 and " + MAX_WORKER_ID);
        }
        this.workerId = workerId;
    }

    // 生成下一个 ID
    public synchronized long nextId() {
        long currentTimestamp = System.currentTimeMillis();

        // 时钟回拨检查
        if (currentTimestamp < lastTimestamp) {
            throw new RuntimeException("Clock moved backwards. Refusing to generate ID.");
        }

        // 如果是同一毫秒内,序列号自增
        if (currentTimestamp == lastTimestamp) {
            sequence = (sequence + 1) & MAX_SEQUENCE;
            // 序列号溢出,等待下一毫秒
            if (sequence == 0) {
                currentTimestamp = waitNextMillis(currentTimestamp);
            }
        } else {
            sequence = 0L; // 新毫秒,序列号重置
        }

        lastTimestamp = currentTimestamp;

        // 组合 ID
        return ((currentTimestamp - START_TIMESTAMP) << TIMESTAMP_SHIFT) |
               (workerId << WORKER_ID_SHIFT) |
               sequence;
    }

    // 等待下一毫秒
    private long waitNextMillis(long currentTimestamp) {
        long timestamp = System.currentTimeMillis();
        while (timestamp <= currentTimestamp) {
            timestamp = System.currentTimeMillis();
        }
        return timestamp;
    }
}
```

#### 2. 在 Spring Boot 中使用
将上述工具类注入到 Spring Boot 的服务中,例如:

```java
import org.springframework.stereotype.Service;

@Service
public class IdGeneratorService {
    private final SnowflakeIdGenerator idGenerator;

    public IdGeneratorService() {
        // 假设 workerId 为 1,可以通过配置动态设置
        this.idGenerator = new SnowflakeIdGenerator(1L);
    }

    public long generateId() {
        return idGenerator.nextId();
    }
}
```

#### 3. 调用示例
在 Controller 中调用服务生成 ID:

```java
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class IdController {
    @Autowired
    private IdGeneratorService idGeneratorService;

    @GetMapping("/generate-id")
    public long generateId() {
        return idGeneratorService.generateId();
    }
}
```

#### 4. 配置 workerId
在分布式系统中,`workerId` 需要唯一,可以通过配置文件或机器标识动态分配。例如,使用 Spring Boot 的 `application.properties`:

```properties
snowflake.worker-id=1
```

然后在服务中读取:

```java
@Service
public class IdGeneratorService {
    private final SnowflakeIdGenerator idGenerator;

    @Autowired
    public IdGeneratorService(@Value("${snowflake.worker-id}") long workerId) {
        this.idGenerator = new SnowflakeIdGenerator(workerId);
    }

    public long generateId() {
        return idGenerator.nextId();
    }
}
```

---

### 使用第三方库
如果你不想自己实现雪花算法,可以使用现成的库,例如:
- **Hutool**:一个流行的 Java 工具库,内置了雪花算法实现。
  ```xml
  <dependency>
      <groupId>cn.hutool</groupId>
      <artifactId>hutool-all</artifactId>
      <version>5.8.11</version>
  </dependency>
  ```
  使用示例:
  ```java
  import cn.hutool.core.lang.Snowflake;

  Snowflake snowflake = new Snowflake(1, 1); // workerId, dataCenterId
  long id = snowflake.nextId();
  ```

- **MyBatis-Plus**:如果你的项目使用 MyBatis-Plus,它也提供了雪花算法的支持。

---

### 注意事项
1. **时钟回拨问题**:如果服务器时间被调整,可能导致 ID 重复。需要在代码中处理时钟回拨。
2. **机器 ID 分配**:在分布式环境中,确保每个节点的 `workerId` 唯一。
3. **性能**:雪花算法适合高并发场景,但序列号耗尽后需等待下一毫秒。

希望这个解答对你有帮助!如果需要更详细的代码或优化建议,请告诉我。

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

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

相关文章

鸿蒙开发之ArkTS联合类型

在鸿蒙开发中&#xff0c;ArkTS是一种基于TypeScript的编程语言&#xff0c;专为鸿蒙应用开发而设计。联合类型&#xff08;Union Types&#xff09;在ArkTS中是一个重要的概念&#xff0c;它允许一个变量存储多种类型的数据&#xff0c;从而增加了代码的灵活性&#xff0c;同时…

K8S学习之基础五十五:k8s中jenkins部署blueOcean

jenkins部署blueOcean 安装插件 BLUE OCEAN 之后会多出一个菜单&#xff0c;可以更详细方便的查看pipeline流程

DeepSeek概述

一、DeepSeek概述 1.1 DeepSeek是什么 DeepSeek是一家专注 通用人工智能&#xff08;AGI&#xff0c;Artificial General Intelligence&#xff09;的中国科技公司&#xff0c;主攻大数据研发与应用。DeepSeek-R1是其开源的推理模型&#xff0c;擅长处理复杂任务且可免费商用…

学习记录(14):iOS部署

时隔多年后&#xff0c;再次部署开发iOS&#x1f601;&#x1f601; 1. Unity端设置&#xff0c;在此不再进行赘述&#xff08;网上一大堆&#xff09; 2. ⚠️&#xff1a;要保证mac比待部署的设备版本要高 3. Xcode: (1) 打开从 Unity 3D 里打包的文件中&#xff0c;找到有…

如何使用DeepSeek编写测试用例?

一、DeepSeek在测试用例设计中的定位 DeepSeek作为AI工具,并非直接替代测试设计,而是通过以下方式提升效率: 快速生成基础用例框架(等价类、边界值等) 智能补充易遗漏场景(如特殊字符、异常流) 自动化脚本片段生成(Python/pytest/JUnit等) 测试数据构造建议(符合业务…

9.4分漏洞!Next.js Middleware鉴权绕过漏洞安全风险通告

今日&#xff0c;亚信安全CERT监控到安全社区研究人员发布安全通告&#xff0c;Next.js 存在一个授权绕过漏洞&#xff0c;编号为 CVE-2025-29927。攻击者可能通过发送精心构造的 x-middleware-subrequest 请求头绕过中间件安全控制&#xff0c;从而在未授权的情况下访问受保护…

【前端】原生项目与框架项目区别

不定期更新&#xff0c;建议关注收藏点赞。 使用 HTML CSS JS 和 Vue 或 React 开发的项目各有其优势与不足&#xff0c;适用于不同的场景。目前基本上都采用框架&#xff0c; 总结 何时选择 HTML CSS JS&#xff1a; 适用于 小型项目、简单静态页面、不需要复杂交互 或 …

WinSCP使用教程:(SFTP、SCP、FTP 和 WebDAV)

WinSCP 是一款免费开源的 Windows 环境下的 SFTP、SCP、FTP 和 WebDAV 客户端&#xff0c;主要用于在本地计算机与远程服务器之间安全地传输文件&#xff0c;并提供基本的文件管理功能。 WinSCP是Windows环境下使用SSH的开源图形化的SFTP的客户端 SSH 的全称是 Secure Shell&…

分布式锁实战:Redis与Redisson的深度解析

一、分布式锁的必要性 在分布式系统中&#xff0c;当多个节点需要对共享资源进行读写操作时&#xff0c;传统的本地锁&#xff08;如Java的synchronized或ReentrantLock&#xff09;无法跨节点生效。此时&#xff0c;必须引入分布式锁来保证操作的原子性和一致性。分布式锁需满…

Dify本地安装部署笔记

目录 方式一【docker安装】&#xff1a; 步骤 1&#xff1a;准备工作 步骤2: 克隆dify仓库 步骤3:部署启动dify 步骤 4&#xff1a;访问 Dify 步骤5:升级dify 方式二【源码安装】&#xff1a; 步骤1. 硬件&#xff1a;最低安装要求 步骤2: 业务服务前的3个服务 1. 安…

质检LIMS系统在食品生产加工企业的应用 如何保证食品生产企业的安全

在食品生产加工领域&#xff0c;质量安全是贯穿全产业链的生命线。随着《食品安全法》对全过程追溯要求的深化&#xff0c;传统实验室管理模式已难以满足高效、精准的质量管控需求。质检实验室信息管理系统&#xff08;LIMS&#xff09;作为数字化升级的核心工具&#xff0c;正…

自动驾驶VLA模型技术解析与模型设计

1.前言 2025年被称为“VLA上车元年”&#xff0c;以视觉语言动作模型&#xff08;Vision-Language-Action Model, VLA&#xff09;为核心的技术范式正在重塑智能驾驶行业。VLA不仅融合了视觉语言模型&#xff08;VLM&#xff09;的感知能力和端到端模型的决策能力&#xff0c;…

UDP套接字编程(代码)

什么是socket套接字编程&#xff1f; 通过Ip地址 端口号这种方式定位一台主机&#xff0c;这样的方式我们就叫做socket套接字。 Udp Socket 接口介绍 这些案列我们使用的接口基本都是一样的&#xff0c;所以在这里我先把接口介绍完&#xff0c;具体的细节后面在说明。 创…

汽车行业可信数据空间研究探索

近期&#xff0c;相关老师在新能源汽车国家大数据联盟微课堂发表了题为“汽车行业可信数据空间研究探索”的演讲&#xff0c;主要包括可信数据空间的概念内涵、汽车行业可信数据空间的发展现状、数据流通场景和技术需求研究、汽车行业可信数据空间的场景建设建议四个方面展开。…

圆弧插补相关算法汇总(C++和ST源代码)

运动控制需要了解相关的插补概念,在阅读本篇博客之前需要了解相关的准备知识,常用链接如下: SMART PLC直线插补详解-CSDN博客文章浏览阅读2.1k次,点赞2次,收藏4次。本文介绍了SMART PLC中轴组对象的概念,详细讲解了直线插补的原理和指令使用,包括SMART PLC从V2.7版本开…

Entity Framework框架

深入理解C#中的Entity Framework框架&#xff1a;从理论到实践 在C#开发中&#xff0c;与数据库交互是几乎所有应用程序的核心需求之一。Entity Framework (EF) 作为微软官方推出的ORM框架&#xff0c;极大地简化了数据库操作。本文将带您深入理解EF框架的核心概念&#xff0c…

C++11QT复习 (五)

文章目录 **Day6-2 成员访问运算符重载&#xff08;2025.03.25&#xff09;****1. 复习****2. 成员访问运算符重载****2.1 箭头运算符 (->) 重载****(1) 语法** **2.2 解引用运算符 (*) 重载****(1) 语法** **3. 代码分析****3.1 代码结构****3.2 代码解析****(1) Data 类**…

简历含金量的描述和注意事项!

背景 最近&#xff0c;在公司负责后端相关面试&#xff0c;简历看了不下 50 份&#xff0c;面试 10&#xff0c;纯手码 2000 多字&#xff0c;说说我对简历的看法&#xff0c;希望给大家一点启发。 教育经历 在众多求职面试中&#xff0c;我发现多数求职者容易忽视教育背景的…

cellnet框架概述

cellnet框架是一个‌高性能、组件化、多协议支持‌的开源服务器网络库&#xff0c;专注于游戏服务器、分布式的多进程通信等场景的开发。 一、核心特性 ‌支持多个主流协议&#xff0c;包括TCP、UDP、HTTP、WebSocket。并且抽象底层协议差异&#xff0c;统一网络连接管理‌。 …

【加密社】如何创建自己的币圈工具站

需要准备的工作 1.域名 2.服务器 周末的时候主要弄了快讯这方面的代码 我这里用的是星球日报的api&#xff0c;也可以订阅他们的rss&#xff0c;这部分在github上是开源的 https://github.com/ODAILY 我这里用的是WordPressonenav主题&#xff0c;然后用小工具在主页展示&am…