JVM内存区域

运行时数据划分为两部分:

  1. 线程私有:虚拟机栈、本地方法栈、程序计数器。
  2. 线程共享:方法区,堆。

运行时数据区域:

img

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区的那个实例化对象

有两种方式,一种使用过句柄指针

img

通过指针直接访问

img

posted @ 2017-02-12 15:32  西北野狼  阅读(205)  评论(0编辑  收藏  举报