小程序教程

java 内存学习笔记 一(内存划分与功能说明)

字号+ 作者:H5之家 来源:H5之家 2017-08-27 16:04 我要评论( )

根据《Java虚拟机规范(Java SE 7版)》的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域: 程序计数器: 程序计数器(Program Counter Register)

根据《Java虚拟机规范(Java SE 7版)》的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域:

这里写图片描述

程序计数器:

程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线
程所执行的字节码的行号指示器。在虚拟机的概念模型里(仅是概念模型,各种虚拟机可能
会通过一些更高效的方式去实现),字节码解释器工作时就是通过改变这个计数器的值来选
取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需
要依赖这个计数器来完成。

由于Java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,
在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)都只会执行一条线
程中的指令。因此,为了线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立
的程序计数器,各条线程之间计数器互不影响,独立存储,我们称这类内存区域为“线程私
有”的内存。

如果线程正在执行的是一个Java方法,这个计数器记录的是正在执行的虚拟机字节码指
令的地址;如果正在执行的是Native方法,这个计数器值则为空(Undefined)。此内存区域
是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。

Java虚拟机栈

与程序计数器一样,Java虚拟机栈(Java Virtual Machine Stacks)也是线程私有的,它的
生命周期与线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行的同时
都会创建一个栈帧(Stack Frame [1] )用于存储局部变量表、操作数栈、动态链接、方法出口
等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出
栈的过程。

经常有人把Java内存区分为堆内存(Heap)和栈内存(Stack),其中栈就是指虚拟机栈.
其中64位长度的long和double类型的数据会占用2个局部变量空间(Slot),其余的数据
类型只占用1个。

在Java虚拟机规范中,对这个区域规定了两种异常状况:如果线程请求的栈深度大于虚
拟机所允许的深度,将抛出StackOverflowError异常;如果虚拟机栈可以动态扩展(当前大部
分的Java虚拟机都可动态扩展,只不过Java虚拟机规范中也允许固定长度的虚拟机栈),如
果扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常.

本地方法栈

本地方法栈(Native Method Stack)与虚拟机栈所发挥的作用是非常相似的,它们之间
的区别不过是虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,而本地方法栈则为虚
拟机使用到的Native方法服务。与虚拟机栈一样,本地方法栈区域也会抛出StackOverflowError
和OutOfMemoryError异常。

Java堆

Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就
是存放对象实例,几乎所有的对象实例都在这里分配内存。这一点在Java虚拟机规范中的描
述是:所有的对象实例以及数组都要在堆上分配 [1] ,但是随着JIT编译器的发展与逃逸分析技
术逐渐成熟,栈上分配、标量替换 [2] 优化技术将会导致一些微妙的变化发生,所有的对象都
分配在堆上也渐渐变得不是那么“绝对”了。

Java堆是垃圾收集器管理的主要区域,因此很多时候也被称做“GC堆”从内存回收的角度来看,由于现在收集器基本都采用分代收集算法,所以Java堆中还可以细分为:新生代和老年代;再细致一点的有Eden空间、From Survivor空间、To Survivor空间等。如果在堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出OutOfMemoryError异常。

*注释 对象实例数据指的是在 Java 中创建的各种实例对象以及它们的值

方法区

方法区(Method Area)与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。它有一个别名叫做Non-Heap(非堆).根据Java虚拟机规范的规定,当方法区无法满足内存分配需求时,将抛出OutOfMemoryError异常。这区域的内存回收目标主要是针对常量池的回收和对类型的卸载

*注释 对象类型数据指的是定义在 Java 代码中的常量、静态变量、以及在类中声明的各种方法、方法字段等等;同事可能包括即时编译器编译后产生的代码数据。

运行时常量池

运行时常量池(Runtime Constant Pool)是方法区的一部分。Class文件中除了有类的版本、
字段、方法、接口等描述信息外,还有一项信息是常量池(Constant Pool Table),用于
存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常
量池中存放。

在Java程序中,有很多的东西是永恒的,不会在运行过程中变化。比如一个类的名字,一个类字段的名字/所属类型,一个类方法的名字/返回类型/参数名与所属类型,一个常量,还有在程序中出现的大量的字面值。
比如下面小段源码中粗体代码显示的部分:

public class ClassTest { private String itemS ="我们 "; private final int itemI =100 ; (String para ){...} }

直接内存
直接内存(Direct Memory)并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规
范中定义的内存区域。本机直接内存的分配不会受到Java堆大小的限制

对象的访问定位

通过句柄访问对象:

这里写图片描述

通过直接指针访问对象:

这里写图片描述

爱编程-编程爱好者经验分享平台

 

1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,请转载时务必注明文章作者和来源,不尊重原创的行为我们将追究责任;3.作者投稿可能会经我们编辑修改或补充。

相关文章
  • 美河制作上海交大最新版JAVA程序设计全

    美河制作上海交大最新版JAVA程序设计全

    2017-08-15 13:00

  • javascript忍者秘籍pdf文字版免费下载

    javascript忍者秘籍pdf文字版免费下载

    2017-06-18 15:06

  • Install Java6 JDK on Ubuntu 12.04

    Install Java6 JDK on Ubuntu 12.04

    2017-04-19 12:02

  • JavaScript高级程序设计

    JavaScript高级程序设计

    2017-04-01 11:03

网友点评