JVM内存区域

来源:互联网 发布:mac wifi管理软件 编辑:程序博客网 时间:2024/06/06 05:02

Java虚拟机所管理的内存包括以下几个运行时数据区域,如下图
这里写图片描述
方法区用于存储已经被虚拟机加载的类信息、常量、静态变量、即使编译器编译后的代码等数据。也有人称其为“永久代”(Permanent Generation),因为HotSpot把GC分代收集扩展至方法区,或者说使用永久代来实现方法区而已。
Java堆是Java虚拟机所管理的内存中最大的一块。此区域的唯一目的就是存放对象实例,几乎所有的对象实例都是在这里分配内存,但是随着JIT编译期的发展与逃逸分析技术逐渐成熟,栈上分配、标量替换优化技术将会导致一些微妙的变化发生,所有对象都分配到堆上也渐渐变得不那么绝对了。
Java堆是垃圾收集器管理的主要区域,因此很多时候也被称为“GC堆”。
Java堆根据垃圾分代收集算法还可以细分为:新生代和老年代。再细致一点的有Eden空间、From Survivor空间、To Survivor空间等。
如果堆中没有内存完成对象实例分配,并且无法再扩展时,将会抛出OutOfMemoryError异常。
Java虚拟机栈描述的是Java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧,用于存储局部变量表、操作数栈、动态链接、方法出口等信息,局部变量表所需的内存空间在编译期间完成分配,这个方法需要在栈帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小;
如果栈深大于虚拟机所允许的深度,将抛出StackOverflowError异常。大部分的虚拟机的可以动态扩容,如果扩容时无法申请到足够的内存,将会抛出OutOfMemoryError异常。
本地方法栈与Java虚拟机栈类似,用于执行native方法,HotSpot虚拟机直接将本地方法栈和Java虚拟机栈合二为一;
PC寄存器(程序计数器)可以看做是当前线程所执行的字节码的行号指示器;

运行时常量池:是方法区的一部分。Class文件中除了有类的版本、字段、方法、接口等描述信息外、还有一项信息是常量池,用于存放编译期产生的各种字面量和符号引用,这部分内容将在类加载后进入方法区 运行时常量池存放。运行时常量池相对于Class文件常量池的另一个重要特征是具备动态性,Java语言并不要求常量一定只有编译期才能产生,也就是并非预置入Class文件中常量池的内容才能进入方法区的运行时常量,运行期间也可能将新的常量放入池中,这种特征被开发人员利用的比较多的便是String类的intern()方法。直接内存:并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域,但这部分内存也被频繁地使用,而且可能导致OutOfMemoryError异常。主要用于NIO作为缓冲区。直接内存的分配不会受到Java堆大小的限制,默认与Java堆最大值一样大,最好是配置一下直接内存的大小。

其中JVM的方法区和堆区由所有线程所共享,其它区域线程隔离。

JVM其实可以理解成一个应用级的OS(操作系统),它的结构可以结合计算机的结构来加以理解,如下图
这里写图片描述

1 0
原创粉丝点击