一步一步学JVM-运行时数据区域

程序计数器(Program Counter Register)

像我们平时读书一样,当我们在去做别的事情之前,我们会对我们读到什么地方了做一个标记,方便我们再回来的时候接着重新读。如果这本书有很多人读呢?那么每个人都会对自己读到了哪里做一个标记。(标记都是个人自己保存)

那么程序计数器也是一样的,同样的代码,可能会被多个线程执行,那么每个线程都要记住自己执行到哪行代码了。这样在线程再次切换执行的时候,知道代码接着从哪里执行。

所以程序计数器是线程私有的,每个线程都会程序计数器。对于正在执行的Native方法,这个计数器值为空。

虚拟机栈(VM Stack)

与程序计数器一样,Java虚拟机栈也是线程私有的。每个线程执行的时候都会创建一个栈帧(Stack Frame)。用来存放局部变量表、操作数栈、动态链接、方法出口等信息。每个方法的调用到执行完成的过程,都对应着一个栈帧在虚拟机中入栈到出栈的过程。

  局部变量表

局部变量表中存放了编译器可知的各种数据类型,和对象引用类型。其中除了64位长度的long和double类型会占用2个局部变量空间(Slot),其余的数据类型只占用1个。所以每个方法需要分配多大的局部变量表空间是完全确定的,在编译期间就完成分配。在方法运行期间不会发生改变。

本地方法栈(Native Method Stack)

与虚拟机栈一样,只是虚拟机栈是为虚拟机执行的Java方法服务,而本地方法栈为虚拟机中使用到的Native方法服务。

Java堆,是被所有线程共享的区域,在虚拟机启动时创建。用来存放对象示例。

  随着对象的创建,Java堆中的对象实例会越来越多,会造成内存溢出。在这些对象中,有百分之八十都是使用完之后就不用了,所以我们可以对这些对象进行回收。就引出了垃圾回收机制,对堆中没用的对象进行清理。为了方便对象清理,又把堆分为了新生代和老年代。其中新生代又可以详细分为Eden区、From Survive区和To Survive区。

方法区

在方法区存储了类的信息,静态变量、常量等数据。虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是它有一个别名Non-Heap(非堆),目的是为了和Java堆区分开。

在HotSpot虚拟机上,方法区也被称为“永久代”,这与Java虚拟机的垃圾回收有关。但是这样设计并不好,更加容易导致内存溢出问题。HotSpot虚拟机现在也有放弃永久代并逐步改为采用Native Memory来实现方法区的规划。从JDK1.7的HotSpot中,已经把原本在永久代的字符串常量池移出。

运行时常量池

运行时常量池也是方法区的一部分。用来存放编译器生成的各种字面量和符号引用。Java语言并不要求常量池一定只有编译器才能产生,运行期间也可能将新的常量放入池中。利用的最多的就是String的intern()方法。

直接内存

直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。但是这部分内存被频繁使用,也可能导致内存溢出异常出现。

时间: 2024-07-31 21:45:54

一步一步学JVM-运行时数据区域的相关文章

JVM【第二回】:【JVM运行时数据区域详解】

上一回对JVM运行时数据区域的组织结构进行了概述,这一回将对各个组成进行详解. 程序计数器[Program Counter Register] 程序计数器是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的行号指示器.在虚拟机的概念模型里,字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支.循环.跳转.异常处理.线程恢复等基础功能都需要依赖这个计数器完成. 由于Java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的

JVM运行时数据区域分析

Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束而建立和销毁. 原文链接:http://sparkyuan.me/2016/04/22/JVM运行时数据区域/ ,转载请注明出去 程序计数器(PC) 程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码行号指示器. - 当前线

JVM 运行时数据区域

Java虚拟机管理的内存包括以下几个运行时数据区域: 1.程序计数器: 程序计数器是一块比较小的内存空间,是当前线程执行的字节码行号指示器.Java多线程是通过线程轮流切换来实现的,所以每个线程都有一个自己独立的程序计数器,各个线程的程序计数器互补干扰. 2.Java虚拟机栈 Java虚拟机栈描述的是Java方法的执行模型:每个方法执行的时候都会创建一个帧栈用于存放局部变量表,操作栈,动态链接,方法出口等信息.一个方法的执行过程,就是这个方法对于帧栈的入栈出栈过程. 3.本地方法栈 本地方法栈与

JVM【第一回】:【JVM运行时数据区域总览】

Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机的进程的启动而存在,有些区域则是依赖用户的线程的启动和结束而建立和销毁.Java虚拟机所管理的内存将会包括一下几个运行时数据区域,如下图所示: 欲知后事如何,且听下回分解

深入理解JVM:JVM运行时数据区域分类

JVM在执行java程序的过程中会把他所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途和创建.销毁时间.有些区域随着虚拟机的启动而存在,有些区域则依赖用户线程的启动和结束而建立和销毁.根据<Java虚拟机规范1.7>规定,Java虚拟机所管理的内存分为以下几个区域: 程序计数器.Java虚拟机栈.本地方法栈.Java堆.方法区.运行时常量池.直接内存 程序计数器 是一块较小的内存空间,他可以看作是当前线程所执行的字节码的行号指示器.在虚拟机的概念模型里,字节码解释器工作时就是通过

JVM调优系列:(二)JVM运行时数据区域

1) Method Area 2) Heap 3) Java Stacks 4) PC Registers 5) Native Method Stacks JAVA的JVM的内存模型大致可分为3个区: 堆区: 1.存储的全部是对象,每个对象都包含一个与之对应的class的信息.(class的目的是得到操作指令) 2.jvm只有一个堆区(heap)被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身 栈区: 线程在执行一个Java方法时候,它的PC寄存器总是指向下一条需要执行的指令, 而它

JVM运行时数据区域解析

Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人想出来. Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束而建立和销毁. 所有线程共享数据区 线程隔离数据区 一.程序计数器(寄存器) 当前线程所执行的字节码行号指示器 字节码解释器工作依赖计数器控制完成 通过执行线程行号记录,让线程轮流切换各条线程

JVM运行时数据区域

程序计数器 1. 线程所执行的字节码的行号指示器: 2.JVM的多线程是通过线程轮流切换并分配处理器执行时间的方式: 3. 每条线程都需要有一个独立的程序计数器,所以是线程私有的内存区域: 4. 执行Java方法,记录的是正在执行的虚拟机字节码指令的地址: 5. 执行Native方法,计数器值为空: 6. 唯一一个在JVM规范中没有规定任何OOM情况的区域: Java虚拟机栈 1. 是线程私有的,其生命周期与线程相同: 2. 是Java方法执行的内存模型: 3. 每个方法执行的同时会创建一个栈帧

轻松认识JVM运行时数据区域(使用思维导图)

下面是个人阅读周志明编写的深入浅出Java虚拟机做成思维导图的笔记,线条.颜色和图片的视觉印象比起单纯文字笔记好得太多了,文字笔记的枯燥以及硬性记忆我就不再多说,特别对于JVM这块略微有点枯燥的知识,更加需要采取更好的方式来认识它.思维导图的模式更加符合大脑认识事物的流程.我将重点的知识抽取出来,又尽量把详细知识描述上去.

思维导图之JVM运行时数据区域

下面是个人阅读周志明编写的深入浅出Java虚拟机做成思维导图的笔记,线条.颜色和图片的视觉印象比起单纯文字笔记好得太多了,文字笔记的枯燥以及硬性记忆我就不再多说,特别对于JVM这块略微有点枯燥的知识,更加需要采取更好的方式来认识它.思维导图的模式更加符合大脑认识事物的流程.我将重点的知识抽取出来,又尽量把详细知识描述上去.