Java直接内存分配和释放方式
2017-08-28 20:33
253 查看
一. 正常分配,回收由GC负责
添加jvm启动参数:-verbose:gc -XX:+PrintGCDetails -XX:MaxDirectMemorySize=40M 循环执行以下代码,可以看到频繁fullGC.ByteBuffer buffer = ByteBuffer.allocateDirect(10 * 1024 * 1024);
当然我也找到一种不需要GC回收由程序员自己回收的方式,不推荐使用
((DirectBuffer)buffer).cleaner().clean();
二. 偏方分配,不安全回收内存由程序员自己负责
如果循环执行下面分配内存代码而不释放会OutOfMemory由于Unsafe是不对外开放的所有使用反射获取theUnsafe属性,第三行f.get(null)能够正确执行的原因是 theUnsafe属性是静态属性。
Field f = Unsafe.class.getDeclaredField("theUnsafe"); f.setAccessible(true); Unsafe unsafe = (Unsafe)f.get(null); long pointer = unsafe.allocateMemory(1024 * 1024 * 20); //释放内存 unsafe.freeMemory(pointer);
原理分析
查看ByteBuffer 源码可知 ByteBuffer.allocateDirect()创建DirectByteBuffer实例,DirectByteBuffer通过Unsafe分配内存,下面具体看一下执行过程。1. 调用
ByteBuffer.allocateDirect(int cap)
2. 创建DirectByteBuffer:主要分三步,第一步调用
Bits.reserveMemory(long size, int cap))在函数内部调用
System.gc()通知GC如有必要进行垃圾回收,第一次调用一般不会触发;第二步,调用
Unsafe.allocateMemory(long var )方法分配内存;第三步,调用
Cleaner.create(Object var0, Runnable var1)创建Cleaner对象,用于回收内存。
3. Cleaner类继承自PhantomReference< Object>在此处保留Cleaner对象的虚引用。此类中还包含一个静态DirectByteBuffer引用队列用于得知那些虚引用所指向的对象已回收,这是一个很棒的设计因为jvm不知道堆外内存的使用情况,通过DirectByteBuffer对象的回收来间接控制堆外内存的回收。
4. 在 2 中
System.gc()给GC一个调用建议,如果在接下来的堆外内存分配中发现空间不足就会触发fullGC 。可以通过XX:MaxDirectMemorySize=40M来模拟。GC之后,“触发”调用
Cleaner.clean()方法,进而调用
Deallocator.run()在run方法中调用
unsafe.freeMemory(long var1)释放堆外内存。
5. 为验证是否因为
System.gc()可在jvm启动参数加入-
XX:+DisableExplicitGC禁用该代码。
6. “触发”阶段,事实上是在Reference类中创建了一个叫Reference Handler的高优先级的守护线程监控着这些“引用”指向的对象。该线程执行Reference类的tryHandlePending方法,判断如对象是Cleaner额外调用clean方法释放内存。
c = r instanceof Cleaner ? (Cleaner) r : null; ................ if (c != null) { c.clean(); return true; }
相关文章推荐
- JAVA的内存分配与释放问题
- java中的参数传递方式以及内存分配情况
- Java内存分配方式
- 最好的方式是用VirtualAlloc分配虚拟内存,它既不是在堆也不是在栈,而是直接在进程的地址空间中保留一块内存
- java中的参数传递方式以及内存分配情况
- JAVA模拟内存分配与释放——首次适应算法
- Java 数组内存分配方式
- C的内存分配释放方式学习
- java中的参数传递方式以及内存分配情况
- C和C++中动态分配和释放内存的方式及其区别
- JAVA的内存分配与释放问题
- 浅析C++内存分配与释放操作过程——三种方式可以分配内存new operator, operator new,placement new
- java内存分配及释放
- C++,java等高级语言的内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区
- java编程思想-内存空间分配(常量池)
- delphi变量内存分配与释放
- [转]Java核心内存分配原理精讲
- Java 内存分配全面浅析转载自http://blog.csdn.net/shimiso
- DLL中用malloc分配了一块内存,但是在exe程序中释放引发的错误:其原因可能是堆被损坏,这也说明 **.exe 中或它所加载的任何 DLL 中有 bug。
- 源码分析:Java对象的内存分配