JAVA虚拟机02---JAVA虚拟机运行时数据区域简介
JAVA虚拟机运行时数据区域
1.程序计数器
1)它可以看做是当前线程执行的字节代码的行指示器,通过改变计数器的值来决定下一步执行的代码
2)它是线程私有的,每个线程都有自己的程序计数器(JAVA中多线程是通过线程轮流切换、分配处理器的执行时间的方式实现的)
3)它的生命周期和线程的生命周期一致
3)对于JVM 来说,程序就是存储在方法区的字节码指令。如果执行的是JAVA方法,程序计数器记录的是正在执行的虚拟机字节码指令的地址,如果是native方法,计数器为空(Underfiend)
4)储蓄计数器内JAVA虚拟机内存区域中在<<JAVA虚拟机规范>>中规定任何OutOfMemoryError的内存区域
2.JAVA虚拟机栈
1)每个方法被执行的时候,都会创建一个栈帧,用于存储局部变量表、操作数栈、动态连接、方法出口等信息(https://www.cnblogs.com/jthr/p/15762205.html)。每一个方法从调用到执行结束的过程就对应着栈帧在虚拟机栈中入栈到出栈的过程
2)它是线程私有的
3)它的生命周期和线程的生命周期一致
4)局部变量表
存放了编译器的可知的各种JAVA虚拟机基本数据类型、对象引用(reference类型,它不等同与对象本身,可能是指向对象起始地址的引用指针或者指向对象的句柄或者其它的与此对象相关的位置[https://www.cnblogs.com/jthr/p/15484760.html 第三块内容])、returnAddress类型(对于 JVM 来说,程序就是存储在方法区的字节码指令,而 returnAddress 类型的值就是指向特定指令内存地址的指针。程序计数器的值就是当前指令所在的内存地址,即 returnAddress 类型的数据(java方法))
这些数据类型在局部变量表中的存储空间以局部变量槽来表示
局部变量表的空间在编译完成分配(一个方法需要在栈帧中分配多大的空间是确定的),在运行期间不会改变局部变量表的大小。这个大小指的是变量槽的数量。至于1个变量槽是占用32个比特、64个比特或者其它,看虚拟机的具体实现
5)虚拟机栈的容量可以动态扩充
6)<<JAVA虚拟机规范>>规定了两种异常。
StandOverfolwError:线程请求的栈的深度大于虚拟机栈的深度。也就是栈帧太多,入不了栈了。
OutOfMemeryError:JAVA虚拟机扩展时无法申请到足够的内存
3.本地方法栈
本地方法栈和虚拟机内存栈基本类似。区别是虚拟机内存栈执行的是JAVA方法,本地方法栈执行的是native方法。
有的虚拟机(HotSpot)把两者合二为一
4.JAVA堆
1)它的唯一目的就是存放对象实例,它是JAVA虚拟机管理的内存中最大的一块
2)它是所有线程共享的,当然,也可以在里面分配出各个线程私有的区域,如本地线程分配换缓冲区TLAB(https://www.cnblogs.com/jthr/p/15484760.html中1.2.2.2)
3)它在虚拟机启动时创建
4)它是垃圾回收器主要管理的内存区域
5)JAVA堆可以处于物理上不连续的内存空间,但在逻辑上是连续的
6)它的大小可以是固定的也可以是可扩展的,主流的虚拟机都是设计的可扩展的
7)但JAVA堆没有内存分配实例且堆无法扩展时,会报出OutOfMemoryError
5.方法区
1)它用于存储被虚拟机加载的类信息(类的版本、字段、方法、接口等)。常量、静态变量、即时编译器编译后的代码缓存等数据。(关于class文件信息https://www.cnblogs.com/jthr/p/15621161.html)
2)它是线程共享的
3)它可以处于物理上不连续的内存空间,但在逻辑上是连续的
4)它的大小可以是固定的也可以是可扩展的,主流的虚拟机都是设计的可扩展的
5)它可以选择不实现垃圾收集。但是一般会进行回收,这个区域的回收目标主要是针对常量池的回收和类型的卸载。
6)它的内存空间无法满足分配需求且无法扩展时,会报出OutOfMemoryError
6.运行时常量池
1)在JDK1.7之前运行时常量池逻辑包含字符串常量池,存放在方法区。
2)在JDK1.7字符串常量池被从方法区拿到了堆中, ,运行时常量池还在方法区,。
3)在JDK1.8 hotspot移除了永久代用元空间(Metaspace)取而代之, 这时候字符串常量池在堆, 运行时常量池还在方法区, 只不过方法区的实现从永久代变成了元空间(Metaspace)
6.1字符串常量池
1)字符串常量池的数据结构
在HotSpot VM里实现的string pool功能的是一个StringTable类,它是一个Hash表,默认值大小长度是1009;StringTable在每个HotSpot VM的实例只有一份,被所有的类共享。字符串常量由一个一个字符组成,放在了StringTable上。
在JDK6.0中,StringTable的长度是固定的,长度就是1009,因此如果放入String Pool中的String非常多,就会造成hash冲突,导致链表过长,当调用String#intern()时会需要到链表上一个一个找,从而导致性能大幅度下降;
在JDK7.0中,StringTable的长度可以通过参数指定:-XX:StringTableSize=66666
2)字符串常量池里存放的内容
在JDK6.0及之前版本中,String Pool里放的都是字符串常量;
在JDK7.0中,由于String#intern()发生了改变,除了字符串常量,String Pool中也可以存放放于堆内的字符串对象的引用
6.2class常量池(Class Constant Pool静态常量池)
1)简介
我们写的每一个Java类被编译后,就会形成一份class文件;class文件中除了包含类的版本、字段、方法、接口等描述信息外,还有一项信息就是常量池表(constant pool table),用于存放编译器生成的各种字面量(Literal)和符号引用(Symbolic References);每个class文件都有一个class常量池。
2)字面量和符号引用
https://www.cnblogs.com/jthr/p/15484424.html
3)静态常量池其实就是class文件里的一部分内容
以下图片是编译后的class文件。
当类加载到内存中后,jvm就会将静态常量池中的内容存放到运行时常量池中
6.3运行时常量池
运行时常量池存在于内存中,也就是静态常量池被加载到内存之后的版本,不同之处是:它的字面量可以动态的添加(String#intern()),符号引用可以被解析为直接引用
JVM在执行某个类的时候,必须经过加载、连接、初始化,而连接又包括验证、准备、解析三个阶段。而当类加载到内存中后,jvm就会将静态常量池中的内容存放到运行时常量池中
运行时常量池是全局共享的,多个类共用一个运行时常量池。并且class文件中常量池多个相同的字符串在运行时常量池只会存在一份
关系小结:
编译好的class文件包含了Constant pool(静态常量池)
class文件被加载时,Constant pool的内容会被加载到运行时常量池
运行时处理池也就包含了类的一些信息(字面量、符号引用等等信息)
在jdk1.7以前,运行时常量池包括字符串常量池,不过1.7开始字符串处理池从运行时处理池移到堆了
JAVA虚拟机02---JAVA虚拟机运行时数据区域简介的更多相关文章
- Java虚拟机 - 结构原理与运行时数据区域
http://liuwangshu.cn/java/jvm/1-runtime-data-area.html 前言 本来计划要写Android内存优化的,觉得有必要在此之前介绍一下Java虚拟机的相关 ...
- Java虚拟机-运行时数据区域
Java虚拟机管理的内存包括如图所示的运行时数据区域: 下面分别进行介绍: 1)程序计数器(Program Counter Register) 占用的内存空间比较小,主要作用就是标识当前线程执行的字节 ...
- 深入理解JAVA虚拟机之JVM性能篇---基础知识点(运行时数据区域)
一. 运行数据区域划分 各个数据区域功能如下: 1. 程序计数器: 较小的一块内存空间,可以看做是当前线程所执行的字节码的行号指示器,每条线程都有一个独立的程序计数器,各条线程之间计数器互不影响,独立 ...
- Java虚拟机一:运行时数据区域
java虚拟机在执行java程序的过程中,会把内存划分为若干个不同的数据区域.每个区域都有各自的用途,创建和销毁时间,按照<java虚拟机规范(Java SE 7 版)>的规定,虚拟机运行 ...
- [二]Java虚拟机 jvm内存结构 运行时数据内存 class文件与jvm内存结构的映射 jvm数据类型 虚拟机栈 方法区 堆 含义
前言简介 class文件是源代码经过编译后的一种平台中立的格式 里面包含了虚拟机运行所需要的所有信息,相当于 JVM的机器语言 JVM全称是Java Virtual Machine ,既然是虚拟机, ...
- Java虚拟机(一)结构原理与运行时数据区域
我们来学习Java虚拟机的结构原理与运行时数据区域. 1.Java虚拟机概述 Oracle官方定义的Java技术体系主要包括以下几个部分: Java程序设计语言 各种平台的Java虚拟机 Class文 ...
- 深入理解Java虚拟机 -- 读书笔记(1):JVM运行时数据区域
深入理解Java虚拟机 -- 读书笔记:JVM运行时数据区域 本文转载:http://blog.csdn.net/jubincn/article/details/8607790 本系列为<深入理 ...
- 《深入理解java虚拟机》笔记(1)运行时数据区域
1.Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的“高墙”,墙外面的人想进去,墙里面的人却想出来. 2.运行时数据区域划分 java虚拟机在执行java程序的过程中会把它所管理的内存划 ...
- 关于Java虚拟机运行时数据区域的总结
Java虚拟机运行时数据区域 程序计数器(Program Counter) 程序计数器作为一个概念模型,这个是用来指示下一条需要执行的字节码指令在哪. Java的多线程实际上是通过线程轮转做到的,如果 ...
- JVM学习(一)Java虚拟机运行时数据区域
一.Java内存区域 1.运行时数据区域 根据<Java 虚拟机规范(Java SE 7 版)>规定,Java 虚拟机所管理的内存包括以下几个运行时数据区域: 1.1 程序计数器 程序计数 ...
随机推荐
- 状态机的技术选型,yyds!
前言 今天跟大家分享一个关于"状态机"的话题.状态属性在我们的现实生活中无处不在.比如电商场景会有一系列的订单状态(待支付.待发货.已发货.超时.关闭):员工提交请假申请会有申请状 ...
- JUC-学习笔记
传统的多线程开发方式 implements Runnable public class oldThreadDev implements Runnable { @Override public void ...
- WEB入门——爆破21-28
WEB21 首先尝试网站后台常见登陆的弱口令,发现错误 则使用burp suite抓包试试看 通过分析,在未填入账号密码时,响应头如下所示: 填入弱口令账号密码,发现响应头如下: 则对应可知账号密 ...
- Qwt开发笔记(二):Qwt基础框架介绍、折线图介绍、折线图Demo以及代码详解
前言 QWT开发笔记系列整理集合,这是目前使用最为广泛的Qt图表类(Qt的QWidget代码方向只有QtCharts,Qwt,QCustomPlot),使用多年,系统性的整理,本系列旨在系统解说并 ...
- Java开发学习(四十八)----MyBatisPlus删除语句之逻辑删除
1.逻辑删除 接下来要讲解是删除中比较重要的一个操作,逻辑删除,先来分析下问题: 这是一个员工和其所签的合同表,关系是一个员工可以签多个合同,是一个一(员工)对多(合同)的表 员工ID为1的张业绩,总 ...
- 论文解读(PCL)《Probabilistic Contrastive Learning for Domain Adaptation》
论文信息 论文标题:Probabilistic Contrastive Learning for Domain Adaptation论文作者:Junjie Li, Yixin Zhang, Zilei ...
- 【QT开发问题】使用自定义的QGroupBox,重写绘图事件paintEvent后边框消失的问题
问题描述 Qt界面开发过程中,使用自定义的QGroupBox,重写绘图事件paintEvent时,出现边框被覆盖的情况,或边框消失的问题. 左图是原始状态,直接重写绘图事件后,会变成右图空白状态. ...
- 论文解读丨【CVPR 2022】不使用人工标注提升文字识别器性能
摘要:本文提出了一种针对文字识别的半监督方法.区别于常见的半监督方法,本文的针对文字识别这类序列识别问题做出了特定的设计. 本文分享自华为云社区<[CVPR 2022] 不使用人工标注提升文字识 ...
- Clickhouse表引擎探究-ReplacingMergeTree
作者:耿宏宇 1 表引擎简述 1.1 官方描述 MergeTree 系列的引擎被设计用于插入极大量的数据到一张表当中.数据可以以数据片段的形式一个接着一个的快速写入,数据片段在后台按照一定的规则进行合 ...
- sniff()函数的总结
作用: sniff()函数主要是用来捕获经过本机网卡的数据包 格式: sniff(filter="",iface="any",prn=function,coun ...