jvm堆外内存(直接内存)

7 篇文章 0 订阅

                                     堆外内存(直接内存)

堆外内存,又被称为直接内存。这部分内存不是由jvm管理和回收的。需要我们手动的回收。

堆内内存是属于jvm的,由jvm进行分配和管理,属于"用户态",而推外内存是由操作系统管理的,属于"内核态"

在jdk1.4中新加入了NIO类,他可以调用native函数库直接分配堆外内存,然后通过java堆中的DirectByteBuffer

对象来指向这块内存,进行内存分配等工作。

可以这样申请堆外内存

package MM;

import java.nio.ByteBuffer;

public class Buffer {

	public static void main(String[] args) {
		// TODO Auto-generated method stub

		while(true) {
		ByteBuffer.allocate(10*1024*1024);
		}
	}

}

可以看到我们一直在申请内存,却一直没有内存溢出。直接内存被释放了。到底堆外内存(直接内存)是怎么释放的呢?

NIO申请直接内存总结:

我们用NIO类申请的内存其实是由jvm进行回收的,并不像unsave那样要我们自己对内存进行管理。这时候系统是不

断回收直接内存的,由NIO申请的直接内存是需要System.gc()来进行内存回收的。系统会帮助我们回收直接内存的。

不过为了提高gc的利用率,我们可能会在代码中加入-XX:+DisableExplicit禁止代码中显示调用gc(System.gc)。采取

并行gc,就是由jvm来自动管理内存回收,而jvm主要是管理堆内内存,也就是当对堆内对象回收的时候,才有可能回

收直接内存,这种不对称性很有可能产生直接内存内存泄漏。

需要注意的是当我们没有指向堆外内存的引用的时候,也会把直接内存回收,这也是上面我们内存没有泄漏的原因。

采用直接内存的优点:

1:对于频繁的io操作,我们需要不断把内存中的对象复制到直接内存。然后由操作系统直接写入磁盘或者读出磁盘。

这时候用到直接内存就减少了堆的内外内存来回复制的操作。

2:我们在运行程序的过程中可能需要新建大量对象,对于一些声明周期比较短的对象,可以采用对象池的方式。但

是对于一些生命周期较长的对象来说,不需要频繁调用gc,为了节省gc的开销,直接内存是必备之选。

3:扩大程序运行的内存,由于jvm申请的内存有限,这时候可以通过堆外内存来扩大内存。

 

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值