JVM内存区域
运行时数据划分为两部分:
- 线程私有:虚拟机栈、本地方法栈、程序计数器。
- 线程共享:方法区,堆。
运行时数据区域:
1 程序计数器:
当前线程所执行的字节码的行号指示器。每条线程都需要有一个独立的程序计数器,各条线程之间互不影响,独立存储,我们称之为线程似有的内存。
如果线程执行的事java方法,计数器记录正在执行的虚拟机字节码指令的地址;
如果执行native方法,程序计数器为空(java虚拟机没有规定outofMemoryError情况的区域)。
2 jVM虚拟机栈:
不存在垃圾回收问题,生命周期和线程一致,线程私有。虚拟机描述java方法执行的内存模型:执行同时会创建栈帧存储局部变量表、操作数栈、动态链接、方法出口等信息。
局部变量表存放:boolean,byte,char,short,int,float,long,double,对象引用,其中64位长度的long,double占用2个局部变量空间,其余的数据占用1个。
虚拟机栈如果线程请求的栈深度大于虚拟机所允许的深度,单线程下抛出stackOverflowError异常;虚拟机栈课动态扩展,但是如果无法申请足够内存就会多线程下抛出outOfMemoryError异常。
3 本地方法栈:
虚拟机栈执行java方法的服务,本地方法栈执行的是native方法。也有可能抛出stackOverflowError和outOfMemoryError异常。
4 java堆:
- 所有线程共享的内存区域,虚拟机启动时创建,存放对象实例。
- java堆是垃圾回收器管理的主要区域,常被成为gc堆。
- 分为新生代,老年代。
- 堆物理上可以不连续,只要逻辑连续即可。
- 堆没有内存完成实例分配的话,堆无法扩展的时候,就会outOfMemoryError。
5 方法区:
线程共享的内存区域,存储虚拟机加载的类信息,常量,静态变量,即时编译后的代码等数据。
6 运行时常量池:
方法区的一部分,常量池存放编译期间生成的字面量和符号引用,在类加载后进入方法区的运行时常量池中存放。
方法池无法申请到内存的时候会抛出outOfMemoryError的异常。
7 直接内存:
不是jvm运行时的数据区一部分,也不是虚拟机规范中的内存区域。
不会受到java堆的限制,但是会受到本机总内存大小限制以及处理器寻址空间的限制。
obj引用怎么你能访问到Java Heap区的那个实例化对象
有两种方式,一种使用过句柄指针
通过指针直接访问