背景
用户咨询了一个java中cpu缓存伪共享场景, 他通过padding多个long 字段隔离 2 个volatile字段,但是实测效果没有提升。
这是个比较有趣的场景,在 jdk8 有更稳定的方案去解决伪共享带来的性能问题。
下面我们展开介绍
- 伪共享问题是什么
- 用户padding方案为何失效
- jdk 的新解法、实现方式和最佳实践
伪共享问题
伪共享(False Sharing)就是多个线程修改位于同一缓存行内的不同变量,导致缓存频繁失效,拖累系统性能。
举个例子:
当两个不相关的变量 A 和 B 恰好落在同一个缓存行时,如果 CPU 核心 1 修改了 A,会导致 CPU 核心 2 的缓存行失效。即使核心 2 只是在操作 B,也必须重新从内存加载数据,这会产生巨大的性能损耗。
引发上面现象的原因是Cache Line,CPU 读取内存时,不是按字节读的,一般是以 64 字节 为单位读入缓存,变量地址很近,会在同一个Cache Line里,哪怕后面的变量不会被当前代码执行,也会被加载。Cache Line大小不同环境会有差异,我们可以用如下命令来确认。
cat /sys/devices/system/cpu/cpu0/cache/index0/coherency_line_size这里只做简述,网上有更详细的图解。
在了解cpu伪共享形成的原因后,解决方法也就很多了。
4. 对象不变,减少不必要的并发。
5. 增加冲突对象的地址距离。
2 的修改比 1 简单,也是最常见的解法,上述用户的修改也是 2 的方式。
用户padding方案失效原因
我把用户的代码做了精简。
public class DataSharingTest { public volatile int m; public volatile long valueA = 0L; public long p1, p2, p3, p4, p5, p6, p7; public volatile long valueB = 0L; public volatile int j; }有并发冲突的是开头的 m 和结尾的 j。他中间加了 long,一个 long 在 java里是 8 字节。中间这么多 long 类型,长度已经超过了 64 字节。
这种写法用户是参考了同事的,并且在他同事那边验证是有效的。
从跑的实际结果上看对象的内存地址是没有分开的。这里被 java 的2 个特性给误导了。
写过c++的同学都经历过计算对象大小的时期,相同的成员变量存在长度不同时,不同的顺序会导致整体对象大小有差异。java 似乎没有要求成员变量顺序,是因为 java自己做了字段重排序,重排成一个最省内存的版本。这就导致了代码的编写和实际运行产生的差异。
我们打出内存对象结构。
com.contended.DataSharingTest object internals: OFF SZ TYPE DESCRIPTION VALUE 0 8 (object header: mark) N/A 8 8 (object header: class) N/A 16 8 long DataSharingTest.valueA N/A 24 8 long DataSharingTest.p1 N/A 32 8 long DataSharingTest.p2 N/A 40 8 long DataSharingTest.p3 N/A 48 8 long DataSharingTest.p4 N/A 56 8 long DataSharingTest.p5 N/A 64 8 long DataSharingTest.p6 N/A 72 8 long DataSharingTest.p7 N/A 80 8 long DataSharingTest.valueB N/A 88 4 int DataSharingTest.m N/A 92 4 int DataSharingTest.j N/A Instance size: 96 bytes Space losses: 0 bytes internal + 0 bytes external = 0 bytes total可以看到 m 和 j 还是排在一起的。类似的代码为什么他同事的是有效的呢,主要来自另外一个特性,指针压缩。
上面object header: class指针压缩时大小只有 4。
com.contended.DataSharingTest object internals: OFF SZ TYPE DESCRIPTION VALUE 0 8 (object header: mark) N/A 8 4 (object header: class) N/A 12 4 int DataSharingTest.m N/A 16 8 long DataSharingTest.valueA N/A 24 8 long DataSharingTest.p1 N/A 32 8 long DataSharingTest.p2 N/A 40 8 long DataSharingTest.p3 N/A 48 8 long DataSharingTest.p4 N/A 56 8 long DataSharingTest.p5 N/A 64 8 long DataSharingTest.p6 N/A 72 8 long DataSharingTest.p7 N/A 80 8 long DataSharingTest.valueB N/A 88 4 int DataSharingTest.j N/A 92 4 (object alignment gap) Instance size: 96 bytes Space losses: 0 bytes internal + 4 bytes external = 4 bytes total这里他同事的最终结果。m 和 j 在内存地址上就是分开的。
java 是同时包含了编译、解释、jit 的语言。使用手动增加变量的方式,弄不好哪个特性或者优化就会导致失效。
jdk的特性和实践
jdk 本身也要编写高并发的库,他也会遇到伪共享问题,在 jdk8中提供了一种稳定的方式来增加内存地址距离。这就是@Contended注解。
jvm虚拟机支持注解
jvm 虚拟机在遇到@Contended注解时会自动增加空白的内存块。
void FieldLayoutBuilder::compute_regular_layout() { bool need_tail_padding = false; prologue(); regular_field_sorting(); if (_is_contended) { _layout->set_start(_layout->last_block()); insert_contended_padding(_layout->start()); need_tail_padding = true; } ... if (!_contended_groups.is_empty()) { for (int i = 0; i < _contended_groups.length(); i++) { FieldGroup* cg = _contended_groups.at(i); LayoutRawBlock* start = _layout->last_block(); insert_contended_padding(start); _layout->add(cg->primitive_fields(), start); _layout->add(cg->oop_fields(), start); need_tail_padding = true; } } }insert_contended_padding就是在加入空白块。
void FieldLayoutBuilder::insert_contended_padding(LayoutRawBlock* slot) { if (ContendedPaddingWidth > 0) { LayoutRawBlock* padding = new LayoutRawBlock(LayoutRawBlock::PADDING, ContendedPaddingWidth); _layout->insert(slot, padding); } }ContendedPaddingWidth就是块的大小。默认为 128。
product(int, ContendedPaddingWidth, 128, \ "How many bytes to pad the fields/classes marked @Contended with")\ range(0, 8192) \ constraint(ContendedPaddingWidthConstraintFunc,AfterErgo)@Contended注解用法
@Contended算是有 3 种用法。
第一种就是加在字段上。
public class Monitoring { @Contended long readCount; @Contended long writeCount; long otherData; }对象内存布局变化如下
OFF SZ TYPE DESCRIPTION VALUE 0 8 (object header: mark) N/A 8 4 (object header: class) N/A 12 4 (alignment/padding gap) 16 8 long Monitoring1.otherData N/A 24 128 (alignment/padding gap) 152 8 long Monitoring1.readCount N/A 160 128 (alignment/padding gap) 288 8 long Monitoring1.writeCount N/A 296 128 (object alignment gap) Instance size: 424 bytes Space losses: 260 bytes internal + 128 bytes external = 388 bytes total加了注解的字段前会加入 128 的内存块。
第二种就是组管理
public class Monitoring { @Contended("stats") long readCount; @Contended("stats") long writeCount; long otherData; // 不在组内 }注解内可以加组名,这样相同组名的变量会放在一起。
OFF SZ TYPE DESCRIPTION VALUE 0 8 (object header: mark) N/A 8 4 (object header: class) N/A 12 4 (alignment/padding gap) 16 8 long Monitoring.otherData N/A 24 128 (alignment/padding gap) 152 8 long Monitoring.readCount N/A 160 8 long Monitoring.writeCount N/A 168 128 (object alignment gap) Instance size: 296 bytes Space losses: 132 bytes internal + 128 bytes external = 260 bytes total这种更利于,每次改动都是多个变量的场景。
第三种是加在类上。
@Contended public class Monitoring2 { long readCount; long writeCount; }这种是作用在每个对象上。适合有对象数组的场景,数组的对象在内存上都是相邻的,通过增加对象的大小,可以保证操作对象之间不会产生影响。
OFF SZ TYPE DESCRIPTION VALUE 0 8 (object header: mark) N/A 8 4 (object header: class) N/A 12 132 (alignment/padding gap) 144 8 long Monitoring2.readCount N/A 152 8 long Monitoring2.writeCount N/A Instance size: 160 bytes Space losses: 132 bytes internal + 0 bytes external = 132 bytes totaljdk里的应用
这里展示一下 jdk 代码里的应用场景
java.lang.Thread把随机种子相关的都放在一个组里,避免了和其他字段的共享。
/** The current seed for a ThreadLocalRandom */ @jdk.internal.vm.annotation.Contended("tlr") long threadLocalRandomSeed; /** Probe hash value; nonzero if threadLocalRandomSeed initialized */ @jdk.internal.vm.annotation.Contended("tlr") int threadLocalRandomProbe; /** Secondary seed isolated from public ThreadLocalRandom sequence */ @jdk.internal.vm.annotation.Contended("tlr") int threadLocalRandomSecondarySeed;java.util.concurrent.Exchanger把 Slot类增加注解,内部是一个Slot[]维护,避免互相干扰。
/** * Padded arena cells to avoid false-sharing memory contention */ @jdk.internal.vm.annotation.Contended static final class Slot { Node entry; } /** * Elimination array; element accesses use emulation of volatile * gets and CAS. */ private final Slot[] arena;最佳实践
上面可以看到jdk.internal.vm.annotation.Contended,这是一个 jdk 内部注解。我们如果引入需要增加
--add-exports=java.base/jdk.internal.vm.annotation=ALL-UNNAMED保证编译和运行通过。jdk 默认是不对用户的模块生效的,我们使用时需要关闭RestrictContended。
-XX:-RestrictContended这种解法本质就是一种拿内存换性能。带来的内存损耗需要仔细评估,否则会带来GC和 OOME。解决的方法有了,我们如何找到比较重要的代码增加注解呢,这里就用到了底层能力。
最直接的发现是 c2c
perf c2c record不过这里需要有内存的事件,不一定有权限。我们可以通过L1-dcache-load-misses来侧面反映。
perf -e L1-dcache-load-misses相关链接