JVM内存区域(运行时数据区)划分
前言:
我们每天都在编写Java代码,编译,执行。很多人已经知道Java源代码文件(.java后缀)会被Java编译器编译为字节码文件(.class后缀),然后由JVM中的类加载器加载各个类的字节码文件,加载完毕之后,交由JVM执行引擎执行。
那在整个程序执行过程中,JVM中怎么存取数据和相关信息呢?
事实上在JVM中是用一段空间来存储程序执行期间需要用到的数据和相关信息,这段空间一般被称作为Runtime Data Area(运行时数据区),也就是我们常说的JVM内存。
一、运行时数据区域包括哪些?
根据《Java虚拟机规范》的规定,运行时数据区通常包括这几个部分:程序计数器(Program Counter Register)、Java虚拟机栈(Java Vitual Machine Stack)、本地方法栈(Native Method Stack)、方法区(Method Area)、堆(Heap)。
二、各个部分存储的信息和负责的职能
1、程序计数器
这个内存区域是Java虚拟机规范中唯一一个没有规定任何OOM(OutOfMemoryError)情况的区域,这是这个区域最大的特点之一,这是因为程序计数器中存储的数据所占空间的大小不会随程序的执行而发生改变,因此,对于程序计数器是不会发生内存溢出现象(OutOfMemory)的。
这个区域主要是负责记录正在执行的虚拟机字节码指令地址,即当前线程执行的字节码的行号指示器(注意:JVM不是直接执行Java代码,而是执行.class文件,所以只要其他编程语言能翻译成.class文件一样能放入JVM中执行)。JVM会给每个线程一个独立的程序计数器,计数器之间互不影响,且通过线程轮流切换并且分配处理器执行时间来实现JVM的多线程。不过当线程执行的是Native方法的时候这个计数器中的值为undefined。
2、Java虚拟机栈
和程序计数器一样的是Java虚拟机栈是线程私有,生命周期和线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行的时候都会创建栈帧,用来存储局部变量表,操作数栈,动态链接,方法出口等信息,每个方法从调用到执行完成的过程,就对应一个栈帧在虚拟机中入栈到出栈的过程,其中64位长度的long和double类型的数据会占用2个局部变量空间,其余的数据类型只占用1个。这里需要理解一下的就是为什么要用栈这个结构呢,比如A方法中调用了B方法,虚拟机中是先让A方法的栈帧进入虚拟机栈执行,当执行到调用B方法的语句就让B栈帧进入,执行完之后B栈帧就出栈,A栈就继续执行。这里注意的是如果递归的方法递归的太深很容易抛出下面两种异常,所以递归虽然写起来方便,但是性能会有所下降,并且容易抛出异常。
Java虚拟机规范中,对这个区域规定了两种异常状况
i. 线程请求栈的深度大于虚拟机所允许栈的深度,将抛出Stack Overflow Error
ii. 如果虚拟机栈可以动态扩展且扩展时无法申请到足够的内存,会抛出OutOfMemoryError
3、本地方法栈
与虚拟机栈作用相似,不过是虚拟机栈为虚拟机执行Java方法提供,而本地方法为虚拟机使用到的Native方法服务,Native方法多是用C++写的。抛出的异常和虚拟机栈相同。
4、Java堆
Java堆是与前面的区域不同的是:这个区域是被所有线程共享的一块内存区域,用来存放对象实例,并为对象实例分配好内存。Java虚拟机规范中这样描述:所有对象实例以及数组都要在堆上分配Java堆也是垃圾收集器管理的主要区域,也叫”GC堆“。由于现在的垃圾回收算法多是分代收集,所以Java堆里面又可分为:新生代和老年代。并且根据Java虚拟机规范的规定:Java堆可以处于物理上不连续的内存空间中,只要逻辑上连续即可。有实例没有被分配,且堆无法再扩展的时候会抛出OutOfMemoryError异常,虚拟机调优其实也主要关注的是这个区域。
5、方法区
与Java堆一样,线程共享,用来存储被虚拟机加载的类信息,常量,静态变量。这个区域Java虚拟机规范对其特别宽松,既可以像Java堆那样不需要连续内存,又可以选择固定大小和可扩展。还可以选择不实现垃圾收集,这个区域的内存回收目标主要是针对常量池的回收和对类型的卸载。当无法满足内存分配需求时,将抛出OutOfMemoryError异常。
目前虚拟机Hotspot已经将这部分存储空间从使用JVM内存换成使用本地内存,即这部分不再叫永久代,而是元空间。这个元空间实际上是JVM动态规定内存大小。这个替换有什么优势呢?因为字符串常量池是存在永久代中,很容易出现性能问题,并且类和方法信息大小难确定,给永久代的的大小指定带来困难,而且GC会对永久代特殊处理,这就增加了GC的复杂性。从JDK1.7开始,字符串常量池就划分进了堆中,其他的更多是元空间在内存划分的算法上更趋于合理
6、运行时常量池
是方法区的一部分。用于存放编译期生成的各种字面量和符号引用,同时也会把翻译出来的直接引用也存储在运行时的常量池中,具有动态性。常量不一定只有编译期才能产生,运行期间也可以将新的常量放入池中。例如String的Intern()方法。同样抛出OutOfMemoryError异常
三、直接内存
这个区域并不是属于运行时数据区域,但是这个区域也会被频繁使用,并且抛出OOM异常。这个区域主要是由于在JDK1.4中新加入了NIO(New Input/Output)类,引入了一种基于通道与缓冲区的I/O方式,它可以使用Native函数库直接分配堆外内存,通过一个储存在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样能避免在Java堆和Native堆中来回复制数据,从而在一些场景中显著提高性能。直接内存分配不会受到Java堆大小的限制,会受到本机总内存大小及处理器寻址空间的限制。会抛出OutOfMemoryError异常
四、总结
只有程序计数器不会报出任何相关OOM异常,而Java虚拟机栈有可能会报出OOM或Stack Overflow异常。Java虚拟机栈主要是存储方法的一些信息,能让方法顺利的执行,而Java堆存储的是对象的信息。虚拟机的垃圾回收算法主要在这一块,并且平常调优的区域也是在这一块。
JVM内存区域(运行时数据区)划分的更多相关文章
- JVM 内存区域 (运行时数据区域)
JVM 内存区域 (运行时数据区域) 链接:https://www.jianshu.com/p/ec479baf4d06 运行时数据区域 Java 虚拟机在执行 Java 程序的过程中会把它所管理的内 ...
- jvm内存模型(运行时数据区)
运行时数据区(runtime data area) jvm定义了几个运行时数据区,这些运行时数据区存储的数据,供开发者的应用或者jvm本身使用.按线程共享与否可以分为线程间共享和线程间独立. 线程间独 ...
- 1、JVM 内存模型+运行时数据区+JVM参数
JMM(内存模型) 1.’主内存+每个线程有自己的内存 JVM运行时数据区 包含:1.程序计算器(每个线程自带):2.JAVA-STACK(每个线程自带):3.本地方法stack:4.堆:5.方法区 ...
- java内存区域----运行时数据区
Java虚拟机的内存区域也叫做java运行时数据区,共分为五个部分:程序计数器,方法区,本地方法栈,虚拟机栈和堆.方法区和堆是线程之间所共有的,程序计数器,本地方法栈,虚拟机栈是线程私有的.其中虚拟机 ...
- JVM内存结构——运行时数据区
在Java虚拟机规范中将Java运行时数据划分为6种,分别为: PC寄存器(程序计数器) Java栈 堆 方法区 运行时常量池 本地方法栈 一.PC寄存器(程序计数器) PC寄存器(Program C ...
- JVM运行时数据区划分
Java内存空间 内存是非常重要的系统资源,是硬盘和cpu的中间仓库及桥梁,承载着操作系统和应用程序的实时运行.JVM内存布局规定了JAVA在运行过程中内存申请.分配.管理的策略,保证了JVM的高效稳 ...
- 深入理解Java虚拟机(一) 运行时数据区划分
前言:从我学Java的第一天开始,我的大学老师就告诉我 Java语言相比C.C++的语言有一个非常强大的功能,那就是自动内存管理:我们用Java编码时不需要申请或释放内存等,这些工作全部交由我们的Ja ...
- JVM学习笔记-运行时数据区
不同于C,C++程序,Java程序的内存管理工作由Java虚拟机(JVM)接管,这减低了java程序员的负担,但如果出现内存泄露与溢出问题如报OutOfMemory,StackOverFlow异常错误 ...
- JVM三部曲之运行时数据区 (第一部)
在接下来的几天想总结下,JVM相关的一些内容,比如下面的这三个内容算是比较核心知识点了 1.运行时数据区域: 在运行时数据区里存储类Class文件元数据(方法区),对象和数组(堆),方法参数局部变量( ...
- [二]Java虚拟机 jvm内存结构 运行时数据内存 class文件与jvm内存结构的映射 jvm数据类型 虚拟机栈 方法区 堆 含义
前言简介 class文件是源代码经过编译后的一种平台中立的格式 里面包含了虚拟机运行所需要的所有信息,相当于 JVM的机器语言 JVM全称是Java Virtual Machine ,既然是虚拟机, ...
随机推荐
- css3纯手写loading效果
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- LC.exe 已退出,代码为-1 问题解决
最近一个c#工程,之前编译正常.后重装系统,安装DevExpress后,编译一直失败,并提示"4>C:\Windows\Microsoft.NET\Framework\v4.0.303 ...
- Ubuntu 16.04 LTS: apt-get update 失败处理 Aborted (core dumped)
在Ubuntu 16.04运行sudo apt-get update出现如下错误: rogn@ubuntu:~$ sudo apt-get update Get:1 http://us.archive ...
- Perl: hash散列转换为Json报错集, perl.c,v $$Revision: 4.0.1.8 $$Date: 1993/02/05 19:39:30 $
bash-2.03$ ./u_json.pl Can't locate object method "encode" via package "JSON" at ...
- 项目:开发->测试->上线:流程规范
1) 项目分析 2) 项目经理分工协作 程序就具体流程: 1: 按项目名称: 建git, 数据库, 线上测试虚拟机 2: 按项目经理分配的大的模块. 自行划分工作阶段 a: 划分为小的模块 b: 预估 ...
- Python之 七级字典查询
# -*- coding:utf- -*- # 作业要求: # 打印直辖市,省,市,县,区,街道五级菜单: # 可以一层一层地进入到所有层 # 可以退出到上一层 # 可随时退出程序 mapChina ...
- UVa 1354 天平难题 (枚举二叉树)
题意: 分析: 其实刚看到这题的时候觉得很难, 以至于结束了第七章然后去做了一遍第六章树的部分.现在再做这题觉得思路并不是太难,因为总共就只有六个结点,那么只要枚举二叉树然后算出天平然后再从叶子往上推 ...
- luogu4093 [HEOI2016/TJOI2016]序列
因为一个变化只会变化一个值,所以 \(dp[i]=max(dp[j])+1,j<i,maxval_j \leq a[i], a[j] \leq minval_i\) 发现跟二维数点问题挺像,树状 ...
- Python接口测试之moco
在现在的软件开发过程中,特别是app的部分,需要的很多数据以及内容,都是来自server端的API,但是不能保证 在客户端开发的时候,api在server端已经开发完成,专门等着前端来调用,理想的情况 ...
- LoadRunner中的参数与变量-产生20位的随机数
LoadRunner中的参数与变量-产生20位的随机数 在LoadRunner脚本开发中,经常会遇到参数与变量相互转换的情况,本文对常见的转换情形进行了方法总结. 1.变量的赋值 //将字符串赋值给变 ...