深入理解Java虚拟机阅读心得(一)
JVM(Java Virtual Machine) 即Java虚拟机,是一种用于计算设备的规范,用于运行Java程序编译后得到的字节码文件(Class文件)
一。JVM的内存区域
1.程序计数器(Programing Counter Register)
用于选取需要JVM执行的字节码指令,最简单的一种方法就是通过修改程序计数器的值来达到选取下一条需要执行的字节码指令的目的。
如在多线程中CPU时间片切换时,可以通过线程私有的程序计数器来知道这次应当执行的是哪一条字节码指令。
每个线程都会有一个独立的程序计数器,即这块内存是属于线程私有(或者说线程隔离)的。若是执行的是Native修饰的方法,则计数器值为空(Undefined)。
此外,程序计数器所在的这块内存区域是唯一一个在JVM规范中没有任何OutOfMemoryError情况的区域
2.Java虚拟机栈(也就是我们常说的栈内存)
是线程私有的。每个方法在执行时都会创建一个栈帧(方法运行时的基础数据结构),用来存储局部变量、动态链接等。
其中,局部变量表中存放:基本数据类型(boolean、byte、char、short、int、long、float、double)和对象引用类型(reference类型,如存储引用的变量?)
在栈帧中,64位长度的long、double都会占用两个局部变量空间(Slot),其余数据类型均只占用1个Slot
虚拟机栈会有两种异常情况:(1)线程请求的栈深度大于虚拟机所允许的深度,会抛出 StackOverflowError 异常。(2)当虚拟机栈可以动态扩展时,若扩展时无法申请到足够内存,则会抛出 OutOfMemoryError 异常
3.Java堆(Java Heap)
是所有线程共享的。同时也是JVM管理内存中最大的一块。JVM规范规定:Java堆可以处于物理上不连续的内存空间,只要逻辑上是连续的即可。
JVM规范同时规定:所有对象实例以及数组都要在堆上分配。即Java堆是用来存储对象实例和数组。
Java堆也是垃圾收集器管理的主要区域,因此有别名"GC堆"(Garbage Collection Heap)。
当Java堆没有内存分配实例对象,且堆再也无法扩展时,会抛出 OutOfMemoryError 异常
4.方法区
是所有线程共享的。JVM规范中将方法区描述为堆的一个逻辑部分,但是却有一个别名叫 Non-Heap。
方法区是用来存储JVM加载的类信息(即类的模板?)、常量、静态变量、即时编译器编译后的代码等数据的。
和Java堆一样不需要连续的内存,且可以选择固定大小或扩展,还可以选择不实现垃圾回收(该区域的回收效率低下,且条件苛刻;但这区域回收是必要的)。
这区域的垃圾回收主要是针对常量池的垃圾回收和对类型的卸载。
当方法区无法满足内存分配需求时,会抛出 OutOfMemoryError 异常。
5.运行时常量池(Runtime Constant Pool)
方法区的一部分。
用于存放编译期生成的各种字面量和符号引用(如字符串常量);具备动态性,运行期间也可能将新的常量放入池中。
String类的intern()方法 ----- 去常量池中寻找当前字符串;若存在,返回找到与当前字符串值一样的字符串地址;若不存在,则将当前字符串存入常量池,再返回该地址。
当常量池无法再申请到内存时,会抛出 OutOfMemoryError 异常。
总结:线程隔离的有:程序计数器,Java虚拟栈。虚拟栈有一个 StackOverflowError 异常
线程共享的有:Java堆,方法区。方法区中有一个运行时常量池。
二。对象的创建
1.虚拟机遇到一条 new 指令时,首先进行类加载检查(是否能在常量池中找到类的符号引用、符号代表的类是否已被加载、解析、初始化等)。
2.通过类加载检查后,在Java堆中分配新内存。分配方式有:(1)指针碰撞:已用内存和空闲内存分置两边,中间设置一个指针作为分界点,每次通过移动指针来分配内存。(2)空闲列表:已使用内存和空闲内存相互交错,JVM维护一个列表,记录哪些内存块是可用的,分配时划分一块足够大的空间给对象。
除了两种分配方式外,考虑到并发情况下的线程安全问题,提出了两种解决方案:
(1)同步处理
(2)本地线程分配缓冲(Thread Local Allocation Buffer, TLAB) : 每个线程在Java堆中预先分配一小块内存,当某线程的TLAB用完后再分配新的TLAB时,才需要同步锁定
3.内存分配完成后,JVM将分配到的内存空间初始化为零值。--- 以保证对象的实例字段可以不赋初始值就能直接使用(这一块没弄懂)
4.对对象进行必要的设置。如对象头中设置对象的哈希码、对象的GC分代年龄、对象是哪个类的实例的标志等
三。对象的内存布局
对象在内存中布局可以分为三块
1.对象头:可分为两部分 :(1) 第一部分用于存储对象自身的运行数据。如哈希码、GC分代年龄、锁状态标志、线程持有的锁等。(2) 第二部分是类型指针,即对象指向它的类元数据的指针。JVM可通过这个指针确定对象是哪个类的实例;但并不是每个JVM都会在对象数据上保留类型指针,因为还有其他方法可以查找对象的元数据。
2.实例数据:对象真正存储的有效信息,即程序代码中定义的各种类型的字段。如父类继承的、子类重定义的等
3.对齐填充:这部分不是必然存在的。仅仅起着占位符的作用。以HotSpot虚拟机为例,对象起始地址必须是8字节的整数倍,即对象大小必须是8字节的整数倍。而对象头部分正好是8字节,当实例数据部分没有对齐时,则需要通过对齐填充来补全。
此外,对象的主流访问定位方式有两种:(1)使用句柄:优点:引用中存储的是稳定的句柄地址,句柄中存放对象实例数据指针和对象类型指针,当对象移动时(垃圾回收时的普遍行为),只需改变句柄中的实例数据指针。(2)直接指针:优点 -- 快,节省时间开销 -- 地址指向的一块内存中既有对象实例数据,也有对象类型数据指针。使用句柄访问时,会在Java堆中划分出一块内存出来作为 句柄池,即句柄池也在Java堆中。
深入理解Java虚拟机阅读心得(一)的更多相关文章
- 深入理解Java虚拟机阅读心得(二)
垃圾收集 程序计数器.虚拟机栈.本地方法栈三个区域随线程而生,随线程而灭:这几个区域的内存分配和回收都具备稳定性,不需要过多的考虑回收的问题.而Java堆和方法区则不一样. Java堆中存储了几乎所有 ...
- 深入理解Java虚拟机阅读心得(三)
Java中提倡的自动内存管理最终可以归结为自动化的解决两个问题: 给对象分配内存 回收分配给对象的内存 先说说回收这一方面的两个主要知识点 一.垃圾收集算法 1.标记-清理算法 首先标记出所有需要回收 ...
- 深入理解Java 虚拟机阅读笔记(一)
1.程序计数器- 占用空间:较小 作用:字节码行号指示器 作用详情:指示指令执行,如(字节码的执行,分支,循环,跳转,异常处理,线程恢复) 特点:线程私有(每个计数器独立计算,上下文相互独立). 2. ...
- 深入理解Java虚拟机--阅读笔记二
垃圾收集器与内存分配策略 一.判断对象是否已死 1.垃圾收集器在对堆进行回收前,要先判断对象是否已死.而判断的算法有引用计数算法和可达性分析算法: 2.引用计数算法是给对象添加引用计数器,有地方引用就 ...
- 深入理解Java虚拟机--阅读笔记一
Java内存区域 一.java运行时数据区域 1. 程序计数器:程序计数器占据的内存空间较小,是当前运行线程执行的字节码的计数:分支.循环.跳转.异常处理.线程恢复等都要依赖技术器来对执行的字节码进行 ...
- 深入理解JAVA虚拟机阅读笔记4——虚拟机类加载机制
虚拟机把描述类的Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 在Java语言中,类型的加载.连接和初始化过程都是 ...
- 深入理解JAVA虚拟机阅读笔记1——JAVA内存区域
一.Java内存区域 1.程序计数器 线程私有. 当前线程所执行的字节码的行号指示器.由于JAVA是多线程的,因此每个线程都独立的程序计数器. 异常:没有规定任何OutOfMemeryError情况的 ...
- 深入理解java虚拟机阅读笔记(1)运行时数据区域
java虚拟机所管理的内存区域主要分为方法区.堆:虚拟机栈.本地方法栈.程序计数器,如图: 1.程序计数器是当前线程所执行的字节码行号指示器,用以记录当前指令执行的位置.程序计数器是线程私有的,每个线 ...
- 深入理解JAVA虚拟机阅读笔记6——线程安全与锁优化
线程安全:如果一个对象可以安全的被多个线程同时使用,那它就是线程安全的. 一.Java中的线程安全 1.不可变 不可变的对象一定是线程安全的.String.枚举类型.java.lang.Number的 ...
随机推荐
- Spring-boot在windows上安装CLI(Command Line Interface)的步骤!
首先去下载安装包,我这里整了一个zip包,一个tar包,下载地址:https://github.com/zhangyawei117/Spring-boot-CLI.git 下载完了之后,把zip包解压 ...
- c语言宏开关 使用
/* * 宏的写法,宏的作用域,当前位置开始到大括号结束,如果没有大括号,到文件结尾. */ #include <stdio.h> //#define HELLO int main(int ...
- rails应用的部署
简单部署 RAILS_ENV=production rake secret /etc/profile export SECRET_KEY_BASE=刚才生成的密钥 source /etc/profil ...
- JS获取form表单数据
以下代码可放在一个js文件中,以便通用: //获取指定表单中指定标签对象 function getElements(formId, label) { var form = document.getEl ...
- PMP学习总结(1) -- 引论
3月18日考试,1个月后出的成绩,当我拿到Pass的结果的时候还是蛮开心的,因为在备考期间,公司项目十分紧急,经常加班到晚上9,10点,而且宝贝女儿也在这个期间出生,所以备考是十分辛苦的,经常晚上11 ...
- 轻松读懂IL
轻松读懂IL先说说学IL有什么用,有人可能觉得这玩意平常写代码又用不上,学了有个卵用.到底有没有卵用呢,暂且也不说什么学了可以看看一些语法糖的实现,或对.net理解更深一点这些虚头巴脑的东西.最重要的 ...
- QEMU KVM libvirt 手册(3) - Storage Media
访问Hard Drive 使用-hda –hdb qemu-system-x86_64 -enable-kvm -name ubuntutest -m 2048 -hda ubuntutest.im ...
- 遇到ANR问题的处理步骤
遇到ANR问题的处理步骤 问题描述 开发中难免会遇到ANR的问题,遇到ANR问题不要想着是因为设备的卡顿出现的问题,我们无法解决,我们应先找到导致ANR的原因,分析原因之后,再来判断这个问题可不可以解 ...
- 吴恩达机器学习笔记43-SVM大边界分类背后的数学(Mathematics Behind Large Margin Classification of SVM)
假设我有两个向量,
- system libzip must be upgraded to version >= 0.11
PHP当前最新版本是PHP7.3,今天在尝试安装的过程中报如下错误: system libzip must be upgraded to version >= 0.11, 根据提示我们可以清楚的 ...