java虚拟机在执行java程序的过程中会吧它所管理的内存划分为若干个不同的数据区:
java虚拟机栈:是线程私有的内存区,生命周期与线程相同。虚拟机栈的描述:每个方法被执行的时候都是都会创建一个栈帧用于存储局部变量表、操作栈>、动态链接、方法出口等信息。每个方法被调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。局部变量表中存放了各种基本数据类型>、对象引用和returnAddress类型。这个区域会出现两个异常:如果线程请求的栈深度大于虚拟机所允许的深度,将会抛出StackOverflowError异常;如果虚拟机栈可
以动态的扩展,当扩展时无法申请到足够的内存时就会抛出OutOfMemoryError异常。
本地方法栈:与虚拟机栈的作用非常类似,区别是虚拟机栈为虚拟机执行java方法服务,而本地方法栈是为虚拟机使用到的Native方法服务。也会抛出StackOverfloeError异常和OutOfMemoryError异常。
java堆:java堆是java虚拟机所管理的内存中最大的一块。java堆是被所有线程共享的一块存储区域。在虚拟机启动时创建,此内存区域的未已目的就是存>放对象实例,几乎所有的对象都在这里分配内存。在java虚拟机中的描述是:所有的对象实例以及数组都要造堆上分配。渐渐变得没有那么“绝对”了。
java堆是GC的主要区域,java堆中还可以细分为:新生代和老年代;再细致一点就是:Eden区、From Survivor空间、To Survivor空间等。java堆可以是物>理上不连续的内存空间,只要逻辑上连续即可。会抛出:OutOfMemoryError异常。
方法区:方法区和堆一样,是各个线程共享的内存区域,他用于存储已被虚拟机家自爱的类信息、常量、静态变量】及时编译器编译后的代码等数据。会抛>出:OutOfMemoryError异常。
运行时常量池:运行时常量池是方法区的一部分。class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译期
生成的各种字面量和符号引用,这部分内同将在类家自爱后存放到方法区的运行时的常量池中。也会抛出:OutOfMemoryError异常。
还有直接内存:直接内存并不是虚拟机运行时数据区的一部分,也不是虚拟机规范中定义的内存区域,但是这部分经常使用也会出现:OutOfMemoryError异>常。
对象访问:
即使最简单的访问也涉及java栈、java堆、方法区单个最重要的内存区域,如下面的代码:
Object obj=new Object();
假设这句代码出现在方法体中object obj这部分的语义将会反应到java栈的本地变量表中作为一个reference类型数据出现。而"new Object()"这部分语义将
会反应到java堆中,形成一块存储了Object类型所有实例数据值的结构化内存,根据具体类型以及虚拟机实现的对象内存布局的不同,这块内存的长度是不固定的,>另外java堆中还包括能查询找到此对象类型数据的地址信息,这类数据则存储于方法区中。不同虚拟机实现的对象访问方式会有所不同,主流的访问方式有两种:使>用句柄和直接指针。
句柄访问方式的最大好处是reference中存储的是稳定的句柄地址,在对象被移动时只会改变句柄中实例数据指针,而reference本身不需要改变。
使用直接指针访问方式的最大好处就是速度更快,节省了一次指针定位的时间开销。Sun HotSpot使用第二种方式进行对象访问。
程序计数器、虚拟机栈、本地方法栈、方法区、堆。
java虚拟机栈:是线程私有的内存区,生命周期与线程相同。虚拟机栈的描述:每个方法被执行的时候都是都会创建一个栈帧用于存储局部变量表、操作栈>、动态链接、方法出口等信息。每个方法被调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。局部变量表中存放了各种基本数据类型>、对象引用和returnAddress类型。这个区域会出现两个异常:如果线程请求的栈深度大于虚拟机所允许的深度,将会抛出StackOverflowError异常;如果虚拟机栈可
以动态的扩展,当扩展时无法申请到足够的内存时就会抛出OutOfMemoryError异常。
本地方法栈:与虚拟机栈的作用非常类似,区别是虚拟机栈为虚拟机执行java方法服务,而本地方法栈是为虚拟机使用到的Native方法服务。也会抛出StackOverfloeError异常和OutOfMemoryError异常。
java堆:java堆是java虚拟机所管理的内存中最大的一块。java堆是被所有线程共享的一块存储区域。在虚拟机启动时创建,此内存区域的未已目的就是存>放对象实例,几乎所有的对象都在这里分配内存。在java虚拟机中的描述是:所有的对象实例以及数组都要造堆上分配。渐渐变得没有那么“绝对”了。
java堆是GC的主要区域,java堆中还可以细分为:新生代和老年代;再细致一点就是:Eden区、From Survivor空间、To Survivor空间等。java堆可以是物>理上不连续的内存空间,只要逻辑上连续即可。会抛出:OutOfMemoryError异常。
方法区:方法区和堆一样,是各个线程共享的内存区域,他用于存储已被虚拟机家自爱的类信息、常量、静态变量】及时编译器编译后的代码等数据。会抛>出:OutOfMemoryError异常。
运行时常量池:运行时常量池是方法区的一部分。class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译期
生成的各种字面量和符号引用,这部分内同将在类家自爱后存放到方法区的运行时的常量池中。也会抛出:OutOfMemoryError异常。
还有直接内存:直接内存并不是虚拟机运行时数据区的一部分,也不是虚拟机规范中定义的内存区域,但是这部分经常使用也会出现:OutOfMemoryError异>常。
对象访问:
即使最简单的访问也涉及java栈、java堆、方法区单个最重要的内存区域,如下面的代码:
Object obj=new Object();
假设这句代码出现在方法体中object obj这部分的语义将会反应到java栈的本地变量表中作为一个reference类型数据出现。而"new Object()"这部分语义将
会反应到java堆中,形成一块存储了Object类型所有实例数据值的结构化内存,根据具体类型以及虚拟机实现的对象内存布局的不同,这块内存的长度是不固定的,>另外java堆中还包括能查询找到此对象类型数据的地址信息,这类数据则存储于方法区中。不同虚拟机实现的对象访问方式会有所不同,主流的访问方式有两种:使>用句柄和直接指针。
使用句柄的访问方式:java堆中将会划分出一块iaineicun作为句柄池,reference中存储的就是对象的句柄地址,而句柄中包含了对象实例数据和类型数据>各自的具体地址信息。如下图所示:
句柄访问方式的最大好处是reference中存储的是稳定的句柄地址,在对象被移动时只会改变句柄中实例数据指针,而reference本身不需要改变。
使用直接指针访问方式的最大好处就是速度更快,节省了一次指针定位的时间开销。Sun HotSpot使用第二种方式进行对象访问。