运行时数据区域:
1、程序计数器:线程执行时的字节码行号指示器。
2、java虚拟机栈(Stack,获者称为方法栈):java方法执行的内存模型。调用方法的时候,就会创建一个虚拟机栈,用于存储局部变量表,操作数栈,动态链接,方法出口等信息。
3、本地方法栈:存放需要调用到的Native方法的相应数据。
以上三者是线程私有的。
下面的两个是线程共享的。
4、java堆(Heap):用于存放实例对象,数组。
5、方法区:用于存放被虚拟机加载的类信息,常量,静态变量,即时编译后的代码数据。
永久代,方法区,元空间:
方法区是虚拟机的规范,HotSpot用方法区来表示永久代。
JDK8中:元空间属于本地内存;存储内容不同,元空间存储类的元信息,静态变量和常量池等并入堆中。
基于分代的思想进行划分:
堆(Heap)的细分:新生代,老年代。
其中,新生代又分为:Eden,From Servivor,To Servivor。
方法区:永久代
分代思想的具体应用:
1、所有的对象,出生在Eden区。
2、eden:from servivor:to servivor = 8 : 1 : 1
3、堆内存不够,发生GC。将存活的对象拷贝到from servivor区域,对象在servivor中每熬过一次GC(from , to 这两个轮流用),就增加一岁。当增加到一定的岁数后,就将对象移到老年代。
新生代发生的GC为Minor GC。发生在老年代GC为Full GC。做一次 Full GC 要比进行一次 Minor GC 的时间更长(10倍)。
内存的分配以及GC:
1、eden创建对象。
2、当eden内存不足时,触发 Minor GC, 将eden中的存活对象,复制到from servivor。
3、当eden内存又不足时,触发 Minor GC,将eden、from servivor中存活的对象,复制到to servivor中。
4、当eden内存又不足时,触发 Minor GC ,将eden、to servivor中存活的对象,复制到from servivor中。
5、在步骤3、4中,执行复制时,会判断对象是否达到进入老年代的条件,如果符合条件,则拷贝到老年代中。
6、如果老年代空间不足,则发生Full GC 。
逃逸分析与栈上分配:
1、如果对象仅在方法体内使用,不进行传递,则该对象就分配到栈上。
java内存模型:
为何需要java内存模型(Java Memory Model,JMM):
用于屏蔽各种操作系统和硬件之间的内存访问差异,使得java能够在各种平台下能够达到一致的运行结果。
java内存模型(虚拟机内存):
1、主内存:所有的变量都存储在主内存。(线程私有的变量,不存在主内存中)
2、工作内存:线程的工作内存保存了该线程使用到的变量的主内存副本拷贝。
两者的工作方式:
线程对变量的所有操作(读取,赋值)都必须在工作内存中进行,而不能直接读写主内存中的变量。不同线程中的工作内存,相互独立,无法直接访问。线程见变量的传递都需要通过主内存来完成。
线程 1<--(直接操作)--> 工作内存 <--(读写)--> 主内存
线程 2<--(直接操作)--> 工作内存 <--(读写)--> 主内存