美文网首页
【JVM】内存结构及对应区域的OOM

【JVM】内存结构及对应区域的OOM

作者: su_wing | 来源:发表于2017-11-06 19:16 被阅读0次

    1.内存结构

    要搞懂JVM内存分为哪几块,每块分别的作用是什么。

    1)程序计数器:是线程私有的,用于记录线程执行代码的行号。

    2)栈:线程私有的,一个线程执行一段代码就会在栈中为该线程开辟一个栈帧,一般用于存放局部变量表(包括基本数据类型及引用变量),操作数栈,方法出口的信息。-Xss

    3)本地方法栈:线程私有的,当执行本地native方法时开辟的空。

    4)堆:是线程共享的,用于存放所有的对象实例。-Xms -Xmx

    5)方法区:线程共享的,存放类信息,静态变量,常量。 -XX:PermSize  -XX:MaxPerSize

    6)运行时常量池:属于方法区的一部分。class文件中有一个常量池,存放的是编译器生成的各个字面量和符号引用,这部分信息在类加载后放在运行时常量池中。

    7)直接内存Direct Memory:不属于运行时数据区,也不属于JVM规范。与NIO相关,NIO可以直接调用Native方法开辟一块堆外内存,用通过一个存储在堆中的DirectByteBuffer对象作为这块内存的引用来进行操作。(可使java程序直接与堆外内存进行通信,而避免了大量的java堆和native堆的来回复制数据)

    需要注意的是,在设置堆大小时,往往会忘了他,导致OOM

    2.OutOfMemeryError异常的发生

    除了程序计数器中,其他存储区域都有可能发生OOM异常。直接趁热打铁,了解一下OOM发生的原因。

    1)java堆溢出

    堆中存放的是对象,不可用的对象会由GC被回收掉,若扔有大量的对象导致空间不足,则会发生OOM。

    先要排查原因,使用内存分析工具Eclipse Memory Analyzer 将堆转存快照dump出来,并进行分析,主要是分析对堆的对象是否有用。

    若对象大部分都无用,则认为是因为内存泄漏,于是我们要使用——来查看GCROOT的引用链是如何引用的,并定位原因。

    若对象确实有用,则认为是内存溢出,则可通过设置-Xmx设置堆最大内存。

    2)虚拟机栈和本地方法栈溢出

    栈中有可能发生两种异常。a.stackoverflow,当线程请求的栈深度超出了最大深度时。b.oom,当栈需要扩展时发现空间不够时。

    可通过-Xss尝试减少栈容量。需要注意的是,-Xss栈空间设置的越大,(被线程瓜分了,每个线程占有资源都很大,数量少)允许的线程数就越少。

    3)方法区溢出

    方法区中主要是类信息,静态变量等,当类过多,静态变量过多,就会发生OOM。

    越来越多的动态代理技术也会产生大量代理类,占用大量空间。

    4)常量池溢出

    当常量池常量过多时,就会发生OOM。可通过String.intern方法产生大量常量并测试。String.intern会先去常量池中找,找到即返回,找不到在常量池中创建一个再返回。

    -XX:PermSize -XX:MaxPermSize用这个设置方法区大小,也可以间接设置常量池大小。

    5)直接内存溢出

    -XX:MaxDirectMemorySize调整,用native方法比如unsafe方法申请堆外内存时,申请过多会OOM.

    直接内存OOM特点:在堆转存快照中看不到明显异常,并且文件很小,程序中又使用了NIO,则可以考虑是直接内存的OOM。

    相关文章

      网友评论

          本文标题:【JVM】内存结构及对应区域的OOM

          本文链接:https://www.haomeiwen.com/subject/fstlmxtx.html