运行时数据区:
概念模型
线程共享区
方法区
存储运行时常量池,已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据
类信息:类的版本,字段,方法,接口
方法区和永久代
垃圾回收在方法区的行为较少,回收效率低 ,(hotspot省去的方法区)
异常的定义,申请过多也会抛出OutOfMemory
Sun/Oracle JDK的HotSpot VM中,直到JDK7都有“持久代”(Permanent Generation,简称PermGen)。也称为方法区。
Oracle JDK8的HotSpot VM去掉“持久代”,以“元数据区”(Metaspace)替代之。
Oracle是在JDK7的时候宣布移除PermGen内存区域,但是知道JDK8才最终移除。
jdk之前:
- 方法区
- Internded String
Oracle是在JDK7的时候宣布移除PermGen内存区域,但是知道JDK8才最终移除。
运行时常量
String s1 = "abc";
String s2 = "abc";
System.out.println(s1 == s2); //true
String s3 = new String("abc");
System.out.println( s1 == s3); //false
System.out.println(s1 == s3.intern()); // true
创建abc会在栈内存,局部变量表 s1 和 s2, 堆内存中"abc" s1 和 s2 都指向了同一个"abc"
常量池是在方法区中,会维护一个StringTable 可以想象为一个hashSet
s3 为new,会重新给其分配一个堆内存 所以s1 == s3 比较地址时是false
inter()是将其加入运行时常量,s1 和s2 可以看做是字节常量,运行时常量拿到字节常量的内存,所以是true
直接内存
NIO直接分配堆外内存
java堆
存储对象的实例
java管理最大的区域
新生代,老年代,Eden空间
-Xmx -Xms 修改参数大小
线程独占区
虚拟机栈
(hotpot 不分虚拟机栈和本地方法栈)
存放方法运行时所需的数据,成为栈帧
描述的是java方法执行的动态内存模型
栈帧:每个方法执行,都会创建一个栈帧,伴随着方法从创建到执行完成,用于存储局部变量表,操作数栈,动态链接,方法出口等
局部变量表: 存放编译器可知的各种基本数据类型,引用类型,returnAddress类型
局部变量表的内存空间是在编译期完成分配,当进入一个方法时,这个方法需要在帧分配多少内存是固定的,在方法运行期间是不会改变局部表的大小
StackOverFlowError:栈内存溢出(递归中常见)
public class StackTest {
private void test(){
System.out.println("方法在执行");
test();
}
public static void main (String[] args){
new StackTest().test();
}
}
本地方法栈
为jvm所调用的native,即本地方法服务
程序计数器
记录当前线程所执行到的字节码的行号
一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器
位于线程独占区
线程执行的是java方法,计数器记录的是正在执行的虚拟机字节码指令的地址,如果执行的是native方法,计数器的值为undefined
唯一一个在java虚拟机规范中没有规定任何OutOfMemoryError情况的区域
----> 执行引擎 ----> 本地库接口 ----> 本地方法区