某个类有以下两个属性
volatile int a;
volatile int b;
int为4字节,两个int为8字节,一个缓存行大小为64字节
故一旦缓存行a失效了,但是b没失效,会连带着b一起失效,因为失效最小的单位就是一个缓存行
这样子会导致效率变低,故解决办法为,让一个属性尽可能地独占一个缓存行
jdk中提供@sun.misc.Contended注解,会对标准了该注解的类或者属性提供额外的字节填充
原子计数器LongAdder源码中的Ceil类就是标注了该注解,防止缓存行伪共享
cpu 修改某共享变量 a 时会先锁定 a 所在的缓存行,并且把其他 cpu 缓存上相关的缓存行设置为无效。
如果被锁定或失效的缓存行里,还存储了其他不相干的变量 b,则会造成不必要的开销,因为其他线程此时会访问不了 b,或者由于缓存行失效需要重新从内存中读取b加载到缓存。
而如果 a 单独使用一个缓存行就可以避免此问题
代码
import lombok.AllArgsConstructor;
import sun.misc.Contended;
import org.openjdk.jol.info.ClassLayout;public class TestAqs {public static void main(String[] args) {@Contended@AllArgsConstructorclass Position {int x;int y;int z;}Position position = new Position(1, 2, 3);String printable = ClassLayout.parseClass(Position.class).toPrintable(position);System.out.println(printable);}
}
现象(添加@sun.misc.Contended)
TestAqs$1Position object internals:OFFSET SIZE TYPE DESCRIPTION VALUE0 4 (object header) 01 00 00 00 (00000001 00000000 00000000 00000000) (1)4 4 (object header) 00 00 00 00 (00000000 00000000 00000000 00000000) (0)8 4 (object header) 28 55 5d ec (00101000 01010101 01011101 11101100) (-329427672)12 4 (object header) d3 01 00 00 (11010011 00000001 00000000 00000000) (467)16 128 (alignment/padding gap) 144 4 int TestAqs$1Position.x 1148 4 int TestAqs$1Position.y 2152 4 int TestAqs$1Position.z 3156 132 (loss due to the next object alignment)
Instance size: 288 bytes
Space losses: 128 bytes internal + 132 bytes external = 260 bytes total
现象(未添加@sun.misc.Contended)
TestAqs$1Position object internals:OFFSET SIZE TYPE DESCRIPTION VALUE0 4 (object header) 01 00 00 00 (00000001 00000000 00000000 00000000) (1)4 4 (object header) 00 00 00 00 (00000000 00000000 00000000 00000000) (0)8 4 (object header) e0 54 3d 49 (11100000 01010100 00111101 01001001) (1228756192)12 4 (object header) 1a 02 00 00 (00011010 00000010 00000000 00000000) (538)16 4 int TestAqs$1Position.x 120 4 int TestAqs$1Position.y 224 4 int TestAqs$1Position.z 328 4 (loss due to the next object alignment)
Instance size: 32 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total
通过以上显现能分析出来,该注解会给该对象首尾各添加128字节
前边儿填充128字节
后边儿填充132字节,填充128字节,由于对象大小必须是8的整数倍,故外加4字节