首页
/ Yitter IdGenerator多线程环境下ID重复问题分析与解决方案

Yitter IdGenerator多线程环境下ID重复问题分析与解决方案

2025-06-28 00:47:45作者:郜逊炳

问题背景

在使用Yitter IdGenerator生成分布式ID时,开发者CupidStar遇到了一个典型的多线程并发问题:在高并发场景下生成的ID出现了重复现象。具体表现为200并发时很容易复现,100多并发时偶尔复现。这个问题直接关系到分布式系统的数据一致性和唯一性保证,值得深入分析。

问题复现

开发者最初使用的测试代码如下:

static {
    IdGeneratorOptions options = new IdGeneratorOptions((short)0);
    YitIdHelper.setIdGenerator(options);
}

public static long getId(){
    return YitIdHelper.nextId();
}

@Test
public void idTest() throws InterruptedException {
    HashSet<Long> set = new HashSet<>();
    int count = 50;
    CountDownLatch countDownLatch = new CountDownLatch(count);
    for (int i = 0; i < count; i++) {
        int finalI = i;
        new Thread(()->{
            for (int j = 0; j < 10; j++) {
                set.add(IdUtil.getId());
            }
            System.out.println("ok"+ finalI);
            countDownLatch.countDown();
        }).start();
    }
    countDownLatch.await();
    System.out.println(set.size());
}

这段代码创建了50个线程,每个线程生成10个ID,理论上应该生成500个唯一ID,但实际测试中出现了重复现象。

问题本质

经过分析,问题并不在于Yitter IdGenerator本身,而是测试代码中存在两个关键问题:

  1. 线程安全问题:使用了非线程安全的HashSet来收集生成的ID,在多线程环境下,HashSet的内部结构可能会被并发修改导致数据丢失或重复。

  2. 测试方法不当:没有正确评估ID生成器的并发性能,测试用例设计存在缺陷。

正确解决方案

开发者最终找到了正确的测试方法,使用线程安全的ConcurrentSkipListSet替代HashSet:

public static void main(String[] args) throws InterruptedException {
    ConcurrentSkipListSet<Long> set = new ConcurrentSkipListSet<>();
    int count = 10;
    CountDownLatch countDownLatch = new CountDownLatch(count);
    IdUtil.getId();
    System.out.println("start");
    long start = System.currentTimeMillis();
    for (int i = 0; i < count; i++) {
        new Thread(()->{
            for (int j = 0; j < 2000; j++) {
                set.add(IdUtil.getId());
            }
            countDownLatch.countDown();
        }).start();
    }
    countDownLatch.await();
    System.out.println(System.currentTimeMillis() - start);
    System.out.println(set.size());
}

这个改进后的测试用例:

  1. 使用线程安全的ConcurrentSkipListSet来收集ID
  2. 增加了测试规模(10个线程,每个生成2000个ID)
  3. 添加了性能统计功能
  4. 移除了不必要的打印语句

深入理解

为什么HashSet在多线程下会出问题

HashSet内部基于HashMap实现,当多个线程同时向HashSet添加元素时,可能会发生:

  1. 扩容时的数据丢失
  2. 链表转红黑树时的结构破坏
  3. 哈希冲突处理不当导致的元素覆盖

ConcurrentSkipListSet的优势

ConcurrentSkipListSet是基于跳表实现的线程安全集合:

  1. 无锁算法实现高并发
  2. 天然有序,便于后续分析
  3. 写操作不会阻塞读操作
  4. 适合高并发场景下的数据收集

最佳实践建议

  1. 测试环境搭建:测试分布式ID生成器时,务必使用线程安全的数据结构来收集结果。

  2. 性能考量:除了验证唯一性,还应该关注ID生成的速度和吞吐量,特别是在高并发场景下。

  3. 异常处理:在实际应用中,应该对ID生成过程添加适当的异常处理和重试机制。

  4. 监控指标:在生产环境中,建议监控ID生成的成功率、耗时等关键指标。

结论

通过这个案例,我们学习到在测试高并发组件时,测试方法本身也需要考虑线程安全性。Yitter IdGenerator作为一款分布式ID生成器,其本身在多线程环境下是安全的,但测试代码如果不当,可能会得出错误的结论。正确使用线程安全的数据结构是验证分布式组件功能的关键前提。

登录后查看全文
热门项目推荐

热门内容推荐

最新内容推荐

项目优选

收起
ohos_react_nativeohos_react_native
React Native鸿蒙化仓库
C++
187
266
openGauss-serveropenGauss-server
openGauss kernel ~ openGauss is an open source relational database management system
C++
138
188
RuoYi-Vue3RuoYi-Vue3
🎉 (RuoYi)官方仓库 基于SpringBoot,Spring Security,JWT,Vue3 & Vite、Element Plus 的前后端分离权限管理系统
Vue
892
529
openHiTLSopenHiTLS
旨在打造算法先进、性能卓越、高效敏捷、安全可靠的密码套件,通过轻量级、可剪裁的软件技术架构满足各行业不同场景的多样化要求,让密码技术应用更简单,同时探索后量子等先进算法创新实践,构建密码前沿技术底座!
C
370
386
Git4ResearchGit4Research
Git4Research旨在构建一个开放、包容、协作的研究社区,让更多人能够参与到科学研究中,共同推动知识的进步。
HTML
19
0
kernelkernel
deepin linux kernel
C
22
6
Cangjie-ExamplesCangjie-Examples
本仓将收集和展示高质量的仓颉示例代码,欢迎大家投稿,让全世界看到您的妙趣设计,也让更多人通过您的编码理解和喜爱仓颉语言。
Cangjie
337
1.11 K
CangjieCommunityCangjieCommunity
为仓颉编程语言开发者打造活跃、开放、高质量的社区环境
Markdown
1.08 K
0
note-gennote-gen
一款跨平台的 Markdown AI 笔记软件,致力于使用 AI 建立记录和写作的桥梁。
TSX
84
4
harmony-utilsharmony-utils
harmony-utils 一款功能丰富且极易上手的HarmonyOS工具库,借助众多实用工具类,致力于助力开发者迅速构建鸿蒙应用。其封装的工具涵盖了APP、设备、屏幕、授权、通知、线程间通信、弹框、吐司、生物认证、用户首选项、拍照、相册、扫码、文件、日志,异常捕获、字符、字符串、数字、集合、日期、随机、base64、加密、解密、JSON等一系列的功能和操作,能够满足各种不同的开发需求。
ArkTS
63
2