《深入理解 Java 虚拟机》学习笔记 -- 内存区域
运行时数据区域
主要分为 6 部分:
- 程序计数器
- 虚拟机栈
- 本地方法栈
- Java 堆
- 方法区
如图所示:
1. 程序计数器(线程私有)
- 程序计数器是当前线程所执行字节码的行号指示器
- 字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令
- 为了线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器
- 执行 Java 方法,计数器记录的是正在执行的虚拟机字节码指令的地址;执行的是 Native 方法,计数器为空
- 程序计数器是唯一一个在 Java 虚拟机中不会出现
OutOfMemoryError
情况的区域
2. Java 虚拟机栈(线程私有)
- 每个方法被执行的时候都会同时创建一个栈帧
- 每一个方法被调用直至执行完成的过程,就对应者一个栈帧再虚拟机中从入栈到出战的过程
- ”栈内存“指的就是虚拟机栈中的局部变量表部分
- 局部变量表存放的是基本数据类型
- 局部变量表所需的内存空间再编译期间完成分配
-
StackOverflowError
: 线程请求的栈深度大于虚拟机所允许的深度 -
OutOfMemoryError
: 虚拟机栈动态扩展无法申请到足够的内存
3. 本地方法栈(线程私有)
- 运行本地方法
- 其他和 Java 虚拟机栈类似
4. Java 堆(线程共享)
- 在虚拟机启动时创建
- 存放对象实例
- 垃圾回收主要区域
5. 方法区(线程共享)
- 存储已被虚拟机加载的类信息,常量,静态变量,即编译器编译后的代码等数据
- 对常量池的回收和对类型的卸载
其他
运行时常量池
运行时常量池是方法区的一部分,
类加载后将 Class 文件中的常量池信息放到方法区的运行时常量池中
-
String 类的
intern()
方法可以把运行期间新的常量放入池中String.intern() 方法的作用:如果池中已经包含一个等于此 String 对象的字符串,则返回代表池中这个字符串的 String 对象;否则,将此 String 对象包含的字符串添加到常量池中,并且返回此 String 对象的引用。
直接内存
- 直接内存并不是虚拟机运行时数据区的一部分,也不是 Java 虚拟机规范中定义的内存区域
对象的创建
步骤
new 类名
根据 new 的参数在常量池中定位一个类的符号引用
如果没有找到这个符号引用,说明类还没有加载,则进行类的加载,解析和初始化
虚拟机为对象分配内存(位于堆中)
将分配的内存初始化为零值(不包括对象头)
这个很好理解,就是基本数据类型初始化为 0,引用类型初始化为 null。
- 调用对象的
<init>
方法
举例
请看下面例子:
public class Test {
public static void main(String args[]) {
String s1 = "abc";
String s2 = "abc";
System.out.println(s1 == s2);
String s3 = new String("abc");
System.out.println(s1 == s3);
System.out.println(s1 == s3.intern());
}
}
打印结果为:
true
false
true
解析:
- 第一个 true 是因为 "abc" 为字符串常量,是放在方法区中的,虽然会开启内存但是会统一指向到运行时常量池中的 HashSet,由 HashSet 统一管理,这时 s1, s2 指向的地址自然就是 HashSet,
s1 == s2
当然为 true - 第二个为 false 是因为 s3 是一个对象实例,实例是放在 Java 堆中,s1 是放在方法区中,两者地址不一样,所以为 false
- 第三个为 true 是因为 intern() 方法可以把新的常量放入池中,所以为 true