1.栈区
2.堆区
堆里容易出现 OutOfMemoryError 错误,并且出现这种错误还会有多种表现形式,比如:
OutOfMemoryError: GC Overhead Limit Exceeded : 当JVM花太多时间执行垃圾回收并且只能回收很少的堆空间时,就会发生此错误。
java.lang.OutOfMemoryError: Java heap space :在创建新的对象时, 堆内存中的空间不足以存放新创建的对象, 就会引发错误。
堆的大小可以通过-Xmx(jvm运行时堆的最大值)和-Xms(jvm运行时堆的最小值)控制。
3.方法区
常量池又分为静态常量池和动态常量池
静态常量池和动态常量池的关系以及区别:
静态常量池存储的是当class文件被java虚拟机加载进来后存放在方法区的一些字面量和符号引用,字面量包括字符串,基本类型的常量,符号引用其实引用的就是常量池里面的字符串,但符号引用不是直接存储字符串,而是存储字符串在常量池里的索引。
动态常量池是当class文件被加载完成后,java虚拟机会将静态常量池里的内容转移到动态常量池里,在静态常量池的符号引用有一部分是会被转变为直接引用,比如类的静态方法或私有方法,实例构造方法,父类方法,这是因为这些方法不能被重写其他版本,所以能在加载的时候就可以将符号引用转变为直接引用,而其他的一些方法是在这个方法被第一次调用的时候才会将符号引用转变为直接引用的。
总结:
方法区里存储着class文件的信息和动态常量池,class文件的信息包括类信息和静态常量池。可以将类的信息看做是对class文件内容的一个框架,里面具体的内容通过常量池来存储。动态常量池里的内容除了是静态常量池里的内容外,还将静态常量池里的符号引用转变为直接引用,而且动态常量池里的内容是能动态添加的。
4. 程序计数器,线程私有
可以看作是当前线程所执行的字节码的行号指示器,字节码解释器工作时就是通过改变整个计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等功能。
由于JVM的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)都只会执行一条线程中的指令。因此为了线程切换后能恢复到正确的执行位置,每条线程都需要一个独立的程序计数器,各条线程之间计数器互不影响,独立存储,所以线程私有。
程序计数器是唯一一个在Java虚拟机规范中没有规定任何OOM(内存溢出)的区域。
5. Java虚拟机栈,线程私有
生命周期和线程相同。虚拟机栈描述的是Java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧。每个方法从调用到执行完成的过程,就对应着一个栈帧在虚拟机中入栈到出栈的过程。
这个区域规定了两种异常情况:
①如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverFlowError
②如果虚拟机在扩展时无法申请到足够的内存空间,则抛出OutOfMemoryError异常
6. 本地方法栈,线程私有
和虚拟机栈的区别就在于一个是为了执行Java方法服务,一个是为了虚拟机使用到的Native方法服务。
注:native方法
native关键字说明其方法是一个原生态方法,方法对应的实现不是在当前文件,而是在用其他语言(如C和C++)实现的文件中,是用做java 和其他语言(如c++)进行协作时使用的,也就是native 后的函数的实现不是用java写的
堆和栈的区别