JVM的内存结构式是什么样的
%
多线程共享
- Java堆
- 方法区:
- 存放已被虚拟机加载的类信息、常量、静态变量、及时编译器编译后的代码等数据。
- 同时包含运行时常量池,用于存放编译期生成的各种字面量和符号引用。
当前线程独享
- Java栈:通常存放基本数据类型和对象的引用;
- 本地方法栈:与虚拟机栈(Java栈)作用很相似,区别是虚拟机栈为虚拟机执行java方法服务,而本地方法栈则是为虚拟机用到的Native方法服务;
- 程序计数器:当前线程所执行指令的行号指示器
内存溢出和内存泄露的区别
%
-
内存溢出:程序运行过程中申请的内存大于系统能够提供的内存,导致无法申请到足够的内存
-
内存泄露:分配内存给临时变量,用完之后却没有被GC回收,始终占用着内存,既不能被使用也不能分配给其他程序,于是就发生了内存泄漏。
-
memory leak会最终会导致out of memory!
内存溢出错误(OOM)的解决办法
%
-
堆溢出
- 原因:java堆用于存储对象实例,我们只要不断的创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,就会在对象数量达到最大堆容量限制后产生内存溢出异常。
- 解决办法:先分清是因为内存泄漏(Memory Leak)还是内存溢出(Memory Overflow)。如果是泄露先用工具判断GC Roots的引用链排查问题,如果不是泄露,检查虚拟机参数:-Xms -
Java栈和本地方法溢出:
- 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常。
- 如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常 -
运行时常量池溢出:
- 如果要向运行时常量池中添加内容,最简单的做法就是使用String.intern()这个Native方法。
- 可以通过命令 -XX:PermSize -XX:MaxPermSize 设置持久代的大小 -
方法区溢出
- 方法区用于存放Class的相关信息,如类名、访问修饰符、常量池、字段描述、方法描述等。
- 个类如果要被垃圾收集器回收,判定条件是很苛刻的。在经常动态生成大量Class的应用中,要特别注意这点。
参考
%