Java中Unsafe数组偏移量计算:为何基址偏移量为16?

Java Unsafe数组元素访问偏移量计算详解

在Java中,使用Unsafe.arrayBaseOffset()Unsafe.arrayIndexScale()方法访问数组元素时,需要进行偏移量计算。其逻辑如下:

首先,根据数组元素大小(以字节为单位),计算索引的左移位数:

int shift = 31 - Integer.numberOfLeadingZeros(scale); // scale = Unsafe.arrayIndexScale()

然后,将索引i左移shift位,得到偏移基数offsetBase

long offsetBase = (long) i << shift;

最后,将偏移基数加上数组的基址偏移量base,得到最终的偏移量offset

long offset = offsetBase + base; // base = Unsafe.arrayBaseOffset()

64位Windows 10系统下基址偏移量为16的解释

在64位Windows 10系统上,未启用-usecompressedoops选项时,Java数组的内存布局通常如下:

  • 8字节: 对象头(Mark Word)
  • 8字节: 类指针(Klass Pointer)
  • 8字节: 元素指针数组(指向数组元素的指针)

由于数组索引从0开始,第一个元素的偏移量即为对象头、类指针和元素指针数组的总大小,也就是16字节。因此,Unsafe.arrayBaseOffset()返回的值为16。