0

0

如何在Java中利用Disruptor实现高性能无锁队列_环形数组RingBuffer的伪共享消除设计

P粉602998670

P粉602998670

发布时间:2026-03-16 16:15:11

|

158人浏览过

|

来源于php中文网

原创

RingBuffer 性能关键在避免伪共享,需用@Contended填充热点字段;publish()前必须完成数据写入,且setCursor()由其自动触发;单生产者应选SingleProducerSequencer以避免冗余内存屏障;waitFor()返回负值是中断信号而非错误。

如何在java中利用disruptor实现高性能无锁队列_环形数组ringbuffer的伪共享消除设计

RingBuffer 为什么必须做缓存行填充(@Contended)

Disruptor 的 RingBuffer 性能关键不在“环形”,而在避免伪共享——多个线程频繁读写同一缓存行(64 字节)时,CPU 缓存一致性协议会反复使其他核心的缓存副本失效,造成严重性能抖动。即使你用的是单生产者单消费者,RingBuffercursorgatingSequences 等字段若挤在同一缓存行,照样被拖垮。

实操建议:

  • Java 8+ 必须开启 -XX:+UseCondensedHeaders 或更关键的是 -XX:-RestrictContended(JDK 9+ 默认开启),否则 @sun.misc.Contended 注解无效
  • Disruptor 3.4.4+ 已在 Sequence 类上加了 @Contended,但自定义的 Event 类如果含 long/int 字段且被多线程访问,仍需手动填充
  • 不要自己写 7 个 long 填充字段——用 PaddingLong 这类现成的填充字段(Disruptor 源码里就有),语义清晰且 JVM 更易优化
  • 验证是否生效:用 JOL(Java Object Layout)工具跑 ClassLayout.parseClass(YourEvent.class).toPrintable(),确认热点字段之间间隔 ≥64 字节

setCursor() 和 publish() 的调用时机与顺序不能颠倒

很多人以为 publish(sequence) 就是“提交”,其实它只是通知消费者:这个位置的数据已就绪;而 setCursor() 是生产者推进自己的写指针。Disruptor 要求严格顺序:先填数据 → 再 publish() → 最后(隐式或显式)setCursor()。错序会导致消费者看到未初始化的事件对象,或死锁在 waitFor()

常见错误现象:

立即学习Java免费学习笔记(深入)”;

  • 消费者拿到的 Event 字段全是默认值(0、null),尤其在高并发下偶发
  • BatchEventProcessor 卡住,sequence.get() == -1 长时间不更新
  • tryNext() 分配成功,但没调 publish() 就 return,后续再无新事件

正确姿势:

long sequence = ringBuffer.next();
try {
    Event event = ringBuffer.get(sequence);
    event.setValue(data); // ✅ 数据写入必须在 publish 前
} finally {
    ringBuffer.publish(sequence); // ✅ publish 是提交动作,不可省略
}
// setCursor() 由 publish() 内部自动完成,无需手动调</p>

<H3>MultiProducerSequencer 和 SingleProducerSequencer 的内存屏障差异</H3>
<p>选错 sequencer 类型不会报错,但会默默引入不必要的 <code>volatile</code> 写和 full memory barrier,吞掉 20%~40% 吞吐量。核心区别在于:单生产者场景下,<code>SingleProducerSequencer</code> 完全不用 CAS 和 volatile 写,只靠 CPU store-store 重排序约束;而 <code>MultiProducerSequencer</code> 每次 <code>next()</code> 都要 CAS 更新 <code>cursor</code>,代价高得多。</p><div class="aritcle_card flexRow">
                                                        <div class="artcardd flexRow">
                                                                <a class="aritcle_card_img" href="/ai/1410" title="Seed-Music"><img
                                                                                src="https://img.php.cn/upload/ai_manual/000/000/000/175680059259508.png" alt="Seed-Music"  onerror="this.onerror='';this.src='/static/lhimages/moren/morentu.png'" ></a>
                                                                <div class="aritcle_card_info flexColumn">
                                                                        <a href="/ai/1410" title="Seed-Music">Seed-Music</a>
                                                                        <p>字节跳动推出的AI音乐生成与编辑工具</p>
                                                                </div>
                                                                <a href="/ai/1410" title="Seed-Music" class="aritcle_card_btn flexRow flexcenter"><b></b><span>下载</span> </a>
                                                        </div>
                                                </div>
<p>使用场景判断要点:</p>
<ul>
<li>一个线程调用 <code>ringBuffer.next()</code> → 用 <code>SingleProducerSequencer</code></li>
<li>多个线程各自持有独立 <code>RingBuffer</code> 实例 → 仍是单生产者,不是多生产者</li>
<li>多个线程共用同一个 <code>RingBuffer</code> 并都调 <code>next()</code> → 才需要 <code>MultiProducerSequencer</code></li>
<li>别被“多消费者”误导:消费者数量不影响 sequencer 类型选择,只影响 gating logic</li>
</ul>
<p>性能影响示例:在 3.2GHz CPU 上,单生产者模式下 <code>next()/publish()</code> 延迟可压到 ~15ns;换成多生产者,直接跳到 ~40ns 以上,且随核数增加恶化</p>

<H3>waitFor() 返回值为负数时的真实含义</H3>
<p><code>waitFor(long sequence, Sequence dependentSequence, SequenceBarrier barrier)</code> 返回负值(如 <code>-1</code>、<code>-2</code>)不是 bug,而是 Disruptor 的中断/超时信号机制。它不抛异常,也不代表失败,只是告诉你“当前等不到,你自己决定下一步”。很多人直接把返回值当序列号用,结果逻辑全乱。</p>
<p>典型误用:</p>
<ul>
<li>把 <code>waitFor()</code> 返回值强制转成 <code>long</code> 当事件序号取数据 → 取到内存垃圾</li>
<li>看到负数就 throw new RuntimeException("timeout") → 中断处理被粗暴覆盖</li>
<li>忽略 <code>dependentSequence</code> 的作用,在多消费者依赖链中导致序列错乱</li>
</ul>
<p>正确响应方式:</p>
<pre class="brush:php;toolbar:false;">try {
    long availableSeq = barrier.waitFor(nextSequence);
    if (availableSeq < 0) {
        // 检查是否中断:Thread.interrupted()
        // 或是否超时:看你的 timeout 逻辑怎么嵌
        continue; // 或 break,取决于业务
    }
    // ✅ 此时 availableSeq 才是安全可用的序列号
    Event event = ringBuffer.get(availableSeq);
    handler.onEvent(event, availableSeq, availableSeq == nextSequence);
    nextSequence = availableSeq + 1;
} catch (AlertException e) {
    // 外部调用了 ringBuffer.alert(),需清理并退出
}

容易被忽略的是:waitFor() 的负返回值具体含义取决于你用的 WaitStrategy 实现——BusySpinWaitStrategy 永远不返回负数,而 TimeoutBlockingWaitStrategy 会在超时时返回 -1。别硬编码判断逻辑。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1132

2024.03.01

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1132

2024.03.01

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1071

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

617

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

335

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

minimax入口地址汇总
minimax入口地址汇总

本专题整合了minimax相关入口合集,阅读专题下面的文章了解更多详细地址。

3

2026.03.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.5万人学习

C# 教程
C# 教程

共94课时 | 11.5万人学习

Java 教程
Java 教程

共578课时 | 83.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号