文档章节

JVM运行时数据区域介绍

加大装益达
 加大装益达
发布于 2017/05/07 17:24
字数 2043
阅读 7
收藏 1

原文链接

此处的内容是根据Java虚拟机规范(Java SE 7)相关内容以及深入理解Java虚拟机等做的总结。可能有不对的地方。了解这些区域,可以从总体上看下虚拟机内部是怎么构造的,网上也有相关的图片介绍,可以适当的记下图片内容,这样可以有一个立体的感受,更容易记忆。

Java虚拟机定义了程序运行期间使用到的运行时数据区域,其中一些与虚拟机生命周期相同,另外一些与线程的生命周期相同。JVM运行时数据区域分为:

  • 程序计数器(Program Counter)
  • Java虚拟机栈(Java Virtual Machine Stack)
  • 堆(Heap)
  • 方法区(Method Area)
  • 本地方法栈(Native Method Stack)

程序计数器(Program Counter)

程序计数器是线程私有的,每条线程都有自己的程序计数器。

Java虚拟机是支持多线程的,多线程是通过线程的轮流切换来实现的,也就是说每次切换都需要在上次停顿的地方重新开始运行,这时候就需要程序计数器来保存当前线程正在执行的字节码指令的地址,切换到该线程的时候,就能知道该执行哪一个字节码指令了。

如果一个线程正在执行的方法是Java方法,程序计数器保存的是Java虚拟机正在执行的字节码指令的地址;如果正在执行的方法是native的,程序计数器的值为undefined。

Java虚拟机栈(Java Virtual Machine Stack)

Java虚拟机栈也是线程私有的,与线程同时创建,用于存储栈帧(Fremas),栈帧用来存储局部变量,操作数栈、指向当前方法所属类的运行时常量池、处理动态链接、方法返回值和异常分派。方法从调用到执行完成的过程就对应着一个栈帧从入栈到出栈的过程。

Java虚拟机栈可以被实现为固定大小的,此时每一条线程的Java虚拟机栈在线程创建的时候容量就已经确定;还可以被实现为根据计算动态扩展和收缩的。

Java虚拟机栈可能会发生异常:

  • 如果线程请求的栈容量超过Java虚拟机栈允许的最大容量,会抛出StackOverflowError异常。
  • 如果虚拟机栈可动态扩展,申请不到足够的内存去完成扩展,或者建立新线程时没有足够的内存去创建虚拟机栈,会抛出OutOfMemoryError异常。

栈帧

栈帧随着方法的调用而创建,随着方法的结束(正常或者异常结束)而销毁,是用来存储数据和部分过程结果的数据结构,同时也被用来处理动态链接(Dynamic Linking)、方法返回值和异常分派(Dispatch Exception)。

栈帧存在于Java虚拟机栈中,栈帧中包含局部变量表、操作数栈和指向当前方法所属类的运行时常量池的引用。

局部变量表和操作数栈的容量在编译期确定,通过方法的Code属性保存并提供给栈帧使用。栈帧的容量大小仅仅取决于Java虚拟机的实现和方法调用时可分配的内存。

局部变量表

局部变量表存在于栈帧中,长度在编译期决定,存储在类和接口的二进制表示中,也就是存储在方法的Code属性中并提供给栈帧使用。

局部变量表可以保存类型为boolean、byte、char、short、int、float、reference、returnAddress,而long和double类型需要两个局部变量表来存储。

局部变量表还用来完成方法调用时参数的传递,一个方法被调用,它的参数会传递至0开始的连续局部变量表位置上。对于实例方法来说,局部变量表第0个位置是用来存储实例方法所在对象的引用,也就是我们通常说的this。

操作数栈

操作数栈存在于栈帧中,是一个LIFO的栈,长度由编译期确定,也是存储在方法的Code属性中提供给栈帧使用。

操作数栈会有一个确定的栈深度,一个long或者double类型的数据会占用两个单位的栈深度,其他数据类型则会占用一个单位深度。

动态链接

栈帧内部包含一个指向运行时常量池的引用(运行时常量池的解释在下面,可以先看一下运行时常量池),这个引用用来支持当前方法的代码实现动态链接。

Class文件中,一个方法调用其他方法或者访问其成员变量是通过符号引用来表示的,动态链接作用就是将符号引用转换为实际的直接引用。

堆(Heap)

堆是各个线程共享的运行时内存区域,也是所有的类实例和数组对象分配内存的区域。堆在虚拟机启动的时候被创建,存储了被垃圾收集器所管理的各种对象。

堆的容量可以是固定大小的,也可以是动态扩展和自动收缩的。Java堆的内存不需要保证是连续的。

Java堆可能发生异常情况:

  • 实际所需的堆超过了最大容量,抛出OutOfMemoryError异常。

方法区(Method Area)

方法区也是被各个线程所共享的运行时内存区域。用于存储类的结构信息,例如运行时常量池、字段、方法数据、构造函数、普通方法的字节码内容、还包括一些在类、实例、接口初始化时用到的特殊方法。

方法区在虚拟机启动的时候被创建,是堆的逻辑组成部分。方法区的容量可以是固定大小的,也可以是动态扩展和自动收缩的。内存空间不需要保证是连续的。

方法区可能发生异常的情况:

  • 方法区的内存不能满足内存分配时,会抛出OutOfMemoryError异常。

运行时常量池

运行时常量池分配在方法区中,类和接口被加载到虚拟机之后,运行时常量池就被创建了。

运行时常量池是类或接口的常量池的运行时表示形式,包括从编译期可知的数值字面量和运行期解析后才能获得的方法或字段引用。

可能会发生异常的情况:

  • 构造运行时常量池所需的内存空间超过了方法区能提供的最大值,会抛出OutOfMemoryError异常。

本地方法栈(Native Method Stack)

用来支持native方法。跟虚拟机栈功能类似。本地方法栈被实现成固定大小或者是动态扩展和收缩的。

可能会发生的异常情况:

  • 如果线程请求的栈容量超过本地方法栈允许的最大容量,会抛出StackOverflowError异常。
  • 如果本地方法栈可动态扩展,申请不到足够的内存去完成扩展,或者建立新线程时没有足够的内存去创建对应的本地方法栈,会抛出OutOfMemoryError异常。

简要总结

程序计数器为线程私有,用来指示程序运行时的位置。

Java虚拟机栈是线程私有的,用来存储局部变量表等,出栈入栈对应着方法的结束开始。

堆是线程共享的区域,虚拟机启动时创建,创建的实例对象和数组都分配在堆上。

方法区是线程共享的区域,虚拟机启动时创建,用来存储类的信息,常量字段等等。

本地方法栈用来执行本地方法的。

© 著作权归作者所有

加大装益达
粉丝 31
博文 83
码字总数 138659
作品 0
浦东
高级程序员
私信 提问
JVM内存结构 VS Java内存模型 VS Java对象模型

Java作为一种面向对象的,跨平台语言,其对象、内存等一直是比较难的知识点。而且很多概念的名称看起来又那么相似,很多人会傻傻分不清楚。比如本文我们要讨论的JVM内存结构、Java内存模型和...

Java架构
2018/07/11
0
0
【JVM】 java内存区域与内存溢出异常

前言 此系列博客是读《深入理解java虚拟机》所做的笔记整理。 No1. JVM内存管理这堵墙? 对C和C++的开发人员来说,在内存管理领域,他们既拥有每一个对象的“所有权”,也担负着每一个对象生...

binggetong
2018/05/07
0
0
JVM(二)Java虚拟机组成详解

导读:详细而深入的总结,是对知识“豁然开朗”之后的“刻骨铭心”,想忘记都难。 Java虚拟机(Java Virtual Machine)下文简称jvm,上一篇我们对jvm有了大体的认识,进入本文之后我们将具体...

王磊的博客
01/14
0
0
JVM 运行时数据区简介及堆与栈的区别

理解JVM运行时的数据区是Java编程中的进阶部分。我们在开发中都遇到过一个很头疼的问题就是OutOfMemoryError(内存溢出错误),但是如果我们了解JVM的内部实现和其运行时的数据区的工作机制,...

大数据之路
2015/08/02
0
1
JVM之Java内存区域与内存溢出异常

Java的JVM可以自动管理内存,包括内存动态分配和垃圾收集等。 简介 JVM在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域都有各自的用途,以及创建和销毁的时间...

Jeffbond
2017/04/08
0
0

没有更多内容

加载失败,请刷新页面

加载更多

通过微服务来正确实施SOA

对于组织来说,能够构建、发展和扩展大型应用程序是至关重要的, 但所涉及的挑战使其成为一项艰巨的任务。正因为如此, 微服务凭借能够将单个组件拆分成围绕特定业务功能的独立服务,已成为构建...

Linux就该这么学
14分钟前
2
0
从 Spark 到 Kubernetes — MaxCompute 的云原生开源生态实践之路

2019年5月14日,喜提浙江省科学技术进步一等奖的 MaxCompute 是阿里巴巴自研的 EB 级大数据计算平台。该平台依托阿里云飞天基础架构,是阿里巴巴在10年前做飞天系统的三大件之分布式计算部分...

阿里云官方博客
17分钟前
1
0
使用python来操作redis用法详解

1、redis连接 redis提供两个类Redis和StrictRedis用于实现Redis的命令,StrictRedis用于实现大部分官方的命令,并使用官方的语法和命令,Redis是StrictRedis的子类,用于向后兼容旧版本的red...

dragon_tech
17分钟前
2
0
给研发工程师的代码质量利器 | SOFAChannel#5 直播整理

> SOFA:Channel,有趣实用的分布式架构频道。 > > 本文根据 SOFAChannel#5 直播分享整理,主题:给研发工程师的代码质量利器 —— 自动化测试框架 SOFAActs。 > > 回顾视频以及 PPT 查看地址...

SOFAStack
20分钟前
1
0
段错误总结

https://blog.csdn.net/e_road_by_u/article/details/61415732 一、段错误是什么 一句话来说,段错误是指访问的内存超出了系统给这个程序所设定的内存空间,例如访问了不存在的内存地址、访问...

悲催的古灵武士
21分钟前
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部