微信公众号: Spark大数据
JVM虚拟机详解
程序计数器
内存空间小,线程私有。字节码解释器工作是就是通过改变这个计数器的值来选取下一条需要执行指令的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖计数器完成。总而言之,程序计数器是为线程间跳转服务的。
如果线程正在执行一个 Java 方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是 Native 方法,这个计数器的值则为 (Undefined)。此内存区域是唯一一个在 Java 虚拟机规范中没有规定任何 OutOfMemoryError 情况的区域。
Java虚拟机栈
线程私有,生命周期和线程一致。描述的是 Java 方法执行的内存模型:每个方法在执行时都会床创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行结束,就对应着一个栈帧从虚拟机栈中入栈到出栈的过程。
局部变量表:存放了编译期可知的八种基本类型、对象引用(reference 类型)和 returnAddress 类型(指向了一条字节码指令的地址)。
Java虚拟机栈可能出现两种类型的异常:线程请求的栈深度大于虚拟机允许的栈深度,将抛出StackOverflowError。虚拟机栈空间可以动态扩展,当动态扩展是无法申请到足够的空间时,抛出OutOfMemory异常。
本地方法栈
它是为本地方法服务的,而本地方法栈则为虚拟机使用到的 Native 方法服务, jdk版本不同,本地方法栈可能不同,有些特殊版本把栈和本地方法栈二合一 。也会有 StackOverflowError 和 OutOfMemoryError 异常。
堆
jvm内存只有一个堆,所有线程共享这个堆,几乎所有的对象都存放在堆里。堆是内存空间最大的一块,相对的运行速度是最慢的。垃圾回收指的就是回收堆里的内存垃圾,所有 new的象都会在堆里。
垃圾回收机制和堆,为什么堆里会有垃圾 ?
当你创建一个对象,这个对象使用完毕之后就是垃圾,当一个对象没有任何一个地址可以指向它的时候,它就是不可达的对象。
jvm 垃圾回收机制回收的就是不可达的对象,目前 jvm 垃圾回收机制用的是科学分析法 ,它会遍历堆里的对象,然后检测是否有地址可达这个对象,如果不可达那么就回收。注意:堆是不连续的空间。
方法区
方法区有两种说法:
1. 被认为是 堆得一部分 负责一部分特殊用途。
2. 和堆是不同概念 也不是一体的。
方法区主要分为三块:
1.静态区,静态的东西都存储在这个地方。
2.类信息区,字节码被加载到内存里 存储在类信息区。
3.常量池,所有的常量存储在常量池里。
什么是常量池 ?
常量池里存储的都是常量,有些类实现了常量池技术,常量是为了减少常见的对象的创建。
什么是常量池技术 ?以 字符串为例:String是不可变的字符串,也就是说 String类的对象本身是不变的,是一个常量在你 String str = "大数据" 时?;崾紫鹊匠A砍刂醒罢?,查看是否有 "大数据" 这个字符串对象。
如果有,把该对象的地址给与 str存储,以后再有变量被赋值为 "大数据"时都会如此操作;如果没有,那么就把 "哈大数据" 对象创建存入常量池,然后地址为str。
注意 :String类实现常量池在显示赋值时支持常量池技术,在new对象时不支持。