&esmp;对于Unsafe的内存操作,我在网上看了很多文章,发现都没怎么写清楚。因为我看完文章,想在内存中设置一个为1的int都做不到,所以我只能自己摸索了。
获取Unsafe
这个是大家都写烂了的,只有两种办法获取,一个是使用反射,一个是使用命令行参数设置类加载器,多了不说了,我直接用反射:
private static Unsafe getUnsafe() {
try {
Field field = Unsafe.class.getDeclaredField("theUnsafe");
field.setAccessible(true);
return (Unsafe) field.get(null);
} catch (Exception e) {
return null;
}
}
测试代码
之所以我设置为1没用,是因为我发现了,分配的内存并没有清空,所以用之前需要把分配的内存全部设置为0,然后再进行使用,否则得到的结果会不对劲,所以我的测试代码如下:
public static void main(String[] args) {
final Unsafe unsafe = getUnsafe();
final long addr = unsafe.allocateMemory(4);
unsafe.setMemory(null, addr, 4, (byte)0);
unsafe.setMemory(null, addr, 1, (byte)1);
final int anInt = unsafe.getInt(addr);
System.out.println(anInt);
System.out.println(Integer.toHexString(anInt));
unsafe.freeMemory(addr);
}
测试结果:
1
1
替代方案
上述操作很麻烦,而且需要手动调用freememory方法进行回收,比较麻烦。可以使用IntBuffer来完成上面例子里的功能,而且JVM会自动回收内存。其内部原理也是unsafe分配堆外内存的。示例代码如下:
public static void main(String[] args) {
final IntBuffer allocate = IntBuffer.allocate(1);
allocate.put(1);
System.out.println(allocate.get(0));
allocate.clear();
}
结果是一样的:
`