一本很久就想读完的书《深入理解Java虚拟机:JVM高级特性与最佳实践》,奈何束之高阁很久,今天翻了出来!
java内存区域与内存溢出异常
1) java虚拟机在执行java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机的
进程启动而存在,有些区域则是依赖用户线程的启动和结束而建立和销毁。
Java虚拟机所管理的内存主要分为:
① 程序计数器(Program Counter Register):是一块较小的内存,作用是可以看做是当前线程所执行的字节码的行号指示器。
由于Java虚拟机的多线程是通过线程的轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)
只会执行一个线程的指令,因此,为了线程切换后能够恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器,各条线程之间的计数器互不影响,
独立存储,称为类内存区域线程私有的内存。
② Java虚拟机栈(Java Virtual Machine Stacks)也是线程私有的,它的生命周期与线程相同。描述的是Java方法执行的内存模型:每个方法被执行的时候都会
同时创建帧栈用于存储局部变量表,操作栈,动态链接,方法出口灯信息。每一个方法被调用直至执行完成的过程,就对应着一个帧栈在虚拟机栈从入栈到出栈
的过程。
局部变量表存放了编译期可知的各种基本数据类型,对象引用和returnAddress类型。long和double的数据会占用2个局部变量空间,其余的只占用一个,当进入
一个方法时,这个方法需要在帧中分配多大的几部变量空间是完全确认的,不会改变其大小。
如果线程请求的栈的深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;
如果虚拟机栈可以动态扩展,当扩展时无法申请到足够的内存时会抛出OutOfMemoryError异常。
示例代码:
public class JavaVMStackOF {
private int stackLength=1;
public void stackLeak()
{
stackLength++;
stackLeak();
}
public static void main(String[] args) {
JavaVMStackOF jvms = new JavaVMStackOF();
try{
jvms.stackLeak();
}catch(Throwable e)
{
System.out.println("stack length:"+jvms.stackLength);
throw e;
}
}
}
③本地方法栈(Native Method Stacks)与虚拟机栈所发挥的作用是非常相似的。其区别不过虚拟机栈为虚拟机执行Java方法服务,而本地方法栈则为虚拟机使用到
的本地Native方法服务。
④Java堆(Java Heap)是Java虚拟机所管理的内存最大的一块。是指被所有线程共享的一块内存区域。在虚拟机启动时创建,此内存的主要目的就是存放对象实例,
几乎所有的对象实例都在这里分配内存。
Java堆是垃圾收集器的主要区域,因此很多时候也被称为“GC堆”。
示例代码:
import java.util.ArrayList;
import java.util.List;
public class HeapOOM {
static class OOMObject{
public static void main(String[] args) {
List<OOMObject> list = new ArrayList<OOMObject>();
while(true)
{
list.add(new OOMObject());
}
}
}
}
⑤方法区(Method Area)与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据。
垃圾收集行为在这个区域较少出现。
⑥运行时常量池(Runtime Constant Pool)是方法区的一部分,Class文件中除了有类的版本,字段,方法,接口等描述信息,还有一项信息是常量池,用于存放
编译期生成的各种字面量和符号引用,这部分内容将在类加载后存放到方法区的运行时的常量池中。当常量池无法申请到内存时会抛出OutOfMemoryError异常。
示例代码:
import java.util.ArrayList;
import java.util.List;
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
// TODO Auto-generated method stub
List<String> list = new ArrayList<String>();
int i=0;
while(true)
{
list.add(String.valueOf(i++).intern());
}
}
}
⑦ 直接内存(Direct Memory)并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域,本机直接内存的分配不会受到Java堆大小的限制,
在配置虚拟机参数是,一般会根据实际内存设置-Xmx等参数信息,但是经常会忽略掉直接内存,使得各个内存区域的总和大于物理内存限制,也会抛出
OutOfMemoryError异常。
其中程序计数器、虚拟机栈、本地方法栈三个区域随线程而生,随线程而灭 ,栈中的栈帧随着方法的进入和退出而有条不紊地执行着出栈和入栈操作。
2)对象访问
Object obj= new Object();
Object obj 这部分的语义将会反应到Java栈的本地变量中,作为reference类型数据出现。
new Object();这部分将会反应反应到Java堆中,形成一块存储Object类型的实例数据值的结构化内存,根据具体类型以及虚拟机实现的对象内存布局的不同,这块内存
长度是不固定的。在Java堆中还必须包含到此对象类型数据的地址信息,这些类型数据则存储在方法区中。
不同的虚拟机实现的对象访问方式不一样,主要有两种:
句柄访问: Java堆中将会划分一块内存来作为句柄池,reference中存储的就是对象的句柄地址,包含了对象实例数据和类型数据各自的具体地址信息。最大的好处就是
reference中存储的稳定的句柄地址,在对象被移动时只会改变句柄中的实例数据指针。而本身不需要移动。
直接访问:Java堆对象的布局中就必须考虑如何防止访问类型数据的相关信息。reference中直接存储的就是对象地址。优势是速度快。节省一次指针定位的时间开销。
由于对象的访问非常的频繁,因此这类开销积少成多也是一项非常可观的执行成本。