1、栈区
public class Math {
public int compute(){//一个方法对应一块栈帧内存区域
int a = l;
int b = 2;
int c = (a + b)*10;
return c;
}
public static void main(String[] args){
Math math = new, Math() ;
math.compute() ;
System.out.println("test");
}
}
栈是先进后出的,main方法运行会先分配一块main方法栈帧内存空间,main方法调用compute()方法,又会给分配一块compute()栈帧内存空间,里面存放着abc局部变量,当方法一结束,局部变量就会被销毁,释放空间出栈,再继续回到main主方法执行
- 一个方法开始运行 栈会给它分配一块栈帧内存区域(入栈)
- 一个方法对应一个栈帧
- 运行结束后会把这块内存销毁(出栈)
一个栈帧中有局部变量,操作数栈,动态连接,方法出口
操作数栈:程序在运行过程中操作数(加减乘除)临时存在的一块数据内存
动态链接:把符号引用(方法名称跟括号)转变为直接引用(方法区中的实际地址)
方法出口:存储返回的信息,比如根据存储的返回信息,从compute()执行结束返回到nain方法执行System.out.println("test"),知道返回到main方法中的哪一行代码继续去执行
2、程序计数器
每一个线程在运行时,java程序计数器会给这个线程分配一块程序计数器内存 来存放当前线程正在或马上要运行的那行代码的位置或行号--->(代码在方法区内部的(内存地址))程序计数器的值会随着代码的运行被字节码执行引擎修改
为什么要设计程序计数器呢?
Java程序是多线程运行的,比如当我们执行线程A的11行代码,突然出现优先级更高的线程B,那么线程A会被挂起,线程B执行结束后继续执行线程A,那么从哪里开始继续运行呢?就从程序计数器中读取到11,继续执行线程A 的11行代码
3、方法区和本地方法栈
方法区在jdk1.8之前叫永久代,jdk1.8之后叫元空间。方法区(元空间):存放常量 、静态变量、 类信息(class字节码文件也叫内元信息)
本地方法栈:本地方法调用时,本地方法栈会给当前线程分配一块栈内存来运行本地方法 (被 native修饰的方法为本地方法)
4、堆
堆主要用于存放各种类的实例对象和数组。在java中被分为两个区域:年轻代和老年代。在java中还有一个永久代的意思,这里最后会单独说明。
默认老年代占整个堆的2/3,年轻代占1/3,新生代又细分为三个区:Eden区、SurvivorFrom、ServivorTo区,三个区的默认比例为:8:1:1。
Eden区:Java新创建的对象绝大部分会分配在Eden区(如果对象太大,则直接分配到老年代)。当Eden区内存不够的时候,就会触发MinorGC(新生代采用的是标记--复制算法),对新生代进行一次垃圾回收。
SurvivorFrom区和To区:在GC开始的时候,对象只会存在于Eden区和名为From的Survivor区,To区是空的,一次MinorGc过后,Eden区和SurvivorFrom区存活的对象会移动到SurvivorTo区中,然后会清空Eden区和SurvivorFrom区,并对存活的对象的年龄+1,如果对象的年龄达到15,则直接分配到老年代。MinorGC完成后,SurvivorFrom区和SurvivorTo区的功能进行互换。下一次MinorGC时,会把SurvivorTo区和Eden区存活的对象放入SurvivorFrom区中,并计算对象存活的年龄。
概念:
- 新生代 GC(Minor GC)——从年轻代空间(包括 Eden 和 Survivor 区域)回收内存被称为 Minor GC,因为 Java 对象大多都具备朝生夕灭的特性,所以 Minor GC 非常频繁,一般回收速度也比较快。
- 老年代 GC(Major GC / Full GC)——指发生在老年代的 GC,出现了 Major GC,经常会伴随至少一次的 Minor GC(但非绝对的,ParallelScavenge 收集器的收集策略里就有直接进行 Major GC 的策略选择过程,HotSpot虚拟机里面关注吞吐量的Parallel Scavenge收集器是基于标记-整理算法的,而关注延迟的CMS收集器则是基于标记-清除算法的
)