php笔记网站,南京城乡建设网站,小型企业网络建设方案,重庆大渡口网站建设解决方案根据 《Java 虚拟机规范》的规定#xff0c;Java 虚拟机所管理的内存将会包括以下截个运行时数据区域#xff0c;如图所示。 1、程序计数器
程序计数器是一块较小的内存空间#xff0c;它可以看做是当前线程所执行的字节码的行号指示器。在 Java 虚拟机的概念模型里#x…根据 《Java 虚拟机规范》的规定Java 虚拟机所管理的内存将会包括以下截个运行时数据区域如图所示。 1、程序计数器
程序计数器是一块较小的内存空间它可以看做是当前线程所执行的字节码的行号指示器。在 Java 虚拟机的概念模型里字节码解释器工作时就是通过该改变这个计数器的值来选取下一条需要执行的字节码指令它是程序控制流的指示器分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。
Java 虚拟机的多线程是通过线程轮流切换、分配处理器执行时间的方式来实现的在任何一个确定的时刻一个处理对于多核处理器来说是一个内核都只会执行一条线程中的指令。因此为了线程切换后能恢复到正确的执行位置每条线程都需要有一个独立的程序计数器各条线程之间的计数器互不影响独立存储我们称这类内存区域为线程私有的内存。 线程执行 Java 方法时计数器值记录的是当前线程正在执行的虚拟机字节码指令的地址线程执行 Native 方法时计数器值为空 这个内存区域是唯一一个在《Java 虚拟机规范》中没有规定任任何 OutOfMemoryError 情况的区域。
2、Java 虚拟机栈
与程序计数器一样Java 虚拟机栈也是线程私有的它的生命周期与线程相同。虚拟机栈描述的是 Java 方法执行的线程内存模型每个方法被执行的时候Java 虚拟机都会同步创建一个栈帧Stack Frame用于存储局部变量表、操作数栈、动态连接、方法出口等信息。每一个方法被调用直至执行完毕的过程就对应着一个栈帧在虚拟机中从入栈到出栈的过程。
局部变量表存放了编译期可知的各种 Java 虚拟机基本数据类型boolean、byte、char、short、int、float、long、double、对象的引用reference 类型它并不同于对象本身可能是一个指向对象起始地址的引用指针也可能是指向一个代表对象的句柄或者其他与此对象相关的位置和 returnAddress 类型指向了一条字节码执行的地址。
这些数据类型在局部变量表中的存储空间以局部变量槽Slot来表示其中 64 位长度的 long 和 double 类型的数据会占用两个变量槽其余的数据类型只占用一个。局部变量表所需的内存空间在编译期间完成分配当进入一个方法时这个方法需要在栈帧中分配多大的局部变量空间是完全确定的在方法运行期间不会改变局部变量表的大小。这里的大小是指变量槽的数量虚拟机真正使用多大的内存空间1 slot 32 bit / 64 bit来实现一个变量槽这是完全由具体的虚拟机实现自行决定的。
在《Java 虚拟机规范中》对这个内存区域规定了两类异常状况如果线程请求的栈深度大于虚拟机所允许的深度将抛出 StackOverflowError 异常如果 Java 虚拟机栈容量可以动态扩展当栈无法扩展时无法申请到足够的内存抛出 OutOfMemoryError 异常。 HotSpot 虚拟机的栈容量是不可以动态扩展的所以该虚拟机是不会由于虚拟机栈无法扩展而导致 OutOfMemoryError 异常。换句话说只要线程申请栈空间成功了就不会有 OOM但是在申请时就失败仍然会出现 OOM 异常的 3、本地方法栈
与虚拟机栈所发挥的作用是非常相似的其区别只是虚拟机栈为虚拟机执行 Java 方法也就是字节码服务而本地方法栈则是为虚拟机使用到的本地Native方法服务。
《Java 虚拟机规范》对本地方法栈中方法使用的语言、使用方式与数据结构并没有任何强制规定因此具体的虚拟机可以根据需要自由实现它甚至有的 Java 虚拟机譬如 HotSpot 虚拟机直接就把本地方法栈和虚拟机栈合二 为一。与虚拟机栈一样本地方法栈也会在栈深度溢出或者栈扩展失败时分别抛出 StackOverflowError 和 OutOfMemoryError 异常。
4、Java 堆
对于 Java 应用程序来说Java 堆是虚拟机所管理的内存中最大的一块。Java 堆是被所有线程共享的一块内存区域在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例Java 世界里几乎所有的对象实例都在这里分配内存。在《Java 虚拟机规范》中对 Java 堆的描述是所有的对象实例以及数组都应当在堆上分配而这里的几乎是指从实现角度来看随着 Java 语言的发展由于即时编译技术的进步尤其是逃逸分析技术的日渐强大栈上分配、标量替换优化手段已经导致一些微妙的变化悄然发生所以说 Java 对象实例都分配在堆上也渐渐变得不是那么绝对了。
Java 堆是垃圾收集器管理的内存区域有时也称为 GC 堆Garbage Collected Heap。从回收内存的角度看由于现代垃圾收集器大部分都是基于分代收集理论设计的所以 Java 堆中经常会出现新生代、老年代、永久代、Eden 空间、From Survivor 空间、To Survivor 空间等名词。这些区域划分仅仅是一部分垃圾收集器的共同特性或者说设计风格而已而非某个 Java 虚拟机具体实现的固有内存布局更不是《Java 虚拟机规范》里对 Java 堆的进一步细致划分。
G1 收集器出现之前作为业界绝对主流的 HotSpot 虚拟机它内部的垃圾收集器全部基于经典分代来设计的需要新生代、老年代收集器搭配才能工作。今天的 HotSpot 里出现了不采用分代设计的新垃圾收集器。 经典分代指新生代、老年代这种划分其中新生代又包含一个 Eden 和两个 Survivor 区 后面的 TLAB 占用 Eden 区默认 1% 从分配内存的角度来看所有线程共享的 Java 堆中可以划分出多个线程私有的分配缓冲区Thread Local Allocation BufferTLAB以提升对象分配时的效率。不过无论从什么角度无论怎么划分都不会改变 Java 堆中存储内容的共性无论是哪个区域存储的都只能是对象的实例将 Java 堆细分的目的只是为了更好地回收内存或者更快地分配内存。
根据《Java 虚拟机规范》的规定Java 堆可以处于物理上不连续的内存空间中但在逻辑上它应该被视为连续的这点就像用磁盘空间去存储文件一样并不要求每个文件都连续存放。但对于大对象如数组对象多数虚拟机是出于实现简单、存储高效的考虑很可能会要求连续的存储空间。
Java 堆既可以被实现成固定大小的也可以是可扩展的不过当前主流的 Java 虚拟机都是按照可扩展来实现的通过参数 -Xmx 和 -Xms 设定。如果在 Java 堆中没有内存完成实例分配并且堆也无法再扩展时Java 虚拟机将会抛出 OutOfMemoryError 异常。
5、方法区
方法区与 Java 堆一样是各个线程共享的内存区域它用于存储已被虚拟机加载的类型信息、常量、静态变量、即时编译后的代码缓存等数据。虽然《Java 虚拟机规范》中把方法区描述为堆的一个逻辑部分但是它却有一个别名叫作非堆Non-Heap目的是与 Java 堆区分开来。
JDK8 以前通常把方法区成为永久代Permanent Generation本质上方法区和永久代并不是等价的因为仅仅是当时的 HotSpot 虚拟机设计团队选择把收集器的分代设计扩展至方法区或者说使用永久代来实现方法区这样使得 HotSpot 的垃圾收集器能够像管理 Java 堆一样管理这部分内存省去专门为方法区编写内存管理代码的工作。但是对于其他虚拟机实现譬如 BEA、JRockit、IBM J9 等来说是不存在永久代的概念的。这种设计导致了 Java 应用更容易遇到内存溢出的问题而且极少数方法例如 String::intern会因永久代的原因而导致不同虚拟机有不同的表现。 永久代有 -XX:MaxPermSize 的上限即使不设置也有默认大小而 J9 和 JRockit 只要没有触碰到进程可用内存的上限例如 32 位系统中的 4GB 限制就不会出问题。 在 JDK6 的时候 HotSpot 开发团队就有放弃永久代逐步改为采用本地内存Native Memory来实现方法区的计划JDK7 的已经把原本放在永久代的字符串常量池和静态变量被移动到了堆中JDK8完全放弃了永久代的概念改用与 J9 和 JRockit 一样在本地内存中实现的元空间Metaspace来代替把 JDK7 中永久代还剩余的内容运行时常量池、类型信息、常量等全部移到元空间中。
《Java 虚拟机规范》对方法区的约束是非常宽松的除了和 Java 堆一样不需要连续的内存和可以选择固定大小或者可扩展外甚至还可以选择不实现垃圾收集。相对而言垃圾收集行为在这个区域的确是较少出现的但并非数据进入了方法区就如永久代的名字一样永久存在了。这个区域的内存回收目标主要是针对常量池的回收和对类型的卸载一般来说这个区域的回收效果比较难令人满意尤其是类型的卸载条件相当苛刻但是这部分区域的回收有时又确实是必要的。根据《Java 虚拟机规范》的规定如果方法区无法满足新的内存分配需求时将抛出 OutOfMemoryError 异常。
6、运行时常量池
运行时常量池Runtime Constant Pool是方法区的一部分。Class 文件中除了有类的版本、字段、方法、接口等描述信息外还有一项信息是常量池表Constant Pool Table用于存放编译期生成的各种字面量与符号引用这部分内容将在类加载后存放到方法区的运行时常量池中。
字面量相当于 Java 语言层面常量的概念
文本字符串final 修饰的成员变量的值基本数据类型的值其他
符号引用属于编译原理方面的概念
类和接口的完全限定名字段名称和描述符方法名称和描述符
Java 虚拟机对于 Class 文件每一部分的格式都有严格规定如每一个字节用于存储哪种数据都必须符合规范上的要求才会被虚拟机认可、加载和执行但对于运行时常量池《Java 虚拟机规范》并没有做任何细节的要求不同提供商实现的虚拟机可以按照自己的需要来实现这个区域不过一般来说除了保存 Class 文件中描述的符号引用外还会把符号引用翻译出来的直接引用也存储在运行时常量池中。
运行时常量池相对于 Class 文件常量池的另一个重要特征是具备动态性Java 语言并不要求常量一定只有编译期才能产生也就是说并非预置入 Class 文件中常量池的内容才能进入方法区运行时常量池运行期间也可以将新的常量放入池中这种特性被开发人员利用得比较多的便是 String 类的 intern 方法。
既然运行时常量池是方法区的一部分自然受到方法区内存的限制当常量池无法再申请到内存时会抛出 OutOfMemoryError 异常。
7、直接内存
直接内存Direct Memory并不是虚拟机运行时数据区的一部分也不是《Java 虚拟机规范》中定义的内存区域。但是这部分内存也被频繁地使用而且也可能导致 OutOfMemoryError 异常。
JDK1.4 中引入了 NIONew Input/Output类引入了一种基于通道Channel与缓冲区Buffer的 I/O 方法它可以使用 Native 函数库直接分配堆外内存然后通过一个存储在 Java 堆里的 DirectByteBuffer 对象作为这块内存的引用进行操作。这样可以在一些场景中显著提高性能因为避免了在 Java 堆和 Native 堆中来回复制数据。
本机直接内存的分配不会受到 Java 堆大小的限制但是会受到本机总内存包括物理内存、SWAP 分区或者分页文件大小以及处理器寻址空间的限制一般服务器管理员配置虚拟机参数时会根据实际内存去设置 -Xmx 等参数信息但经常忽略掉直接内存使得各个内存区域总和大于物理内存限制包括物理的和操作系统级的限制从而导致动态扩展时出现 OutOfMemoryError 异常。