Java虚拟机 学习笔记

资料来源:https://frank-lam.github.io/fullstack-tutorial/#/JavaArchitecture/05-Java%E8%99%9A%E6%8B%9F%E6%9C%BA

 


JVM体系结构

 

Java 的开发遵循 write once run anywhere(“一次编写到处乱跑”)理念,它运行在 VM(虚拟机)上。编译器将 Java 文件编译成 Java.class 文件,之后,将 .class 文件输入到 JVM 中,加载并执行该类文件。

以下为体系结构:

 

 

JVM模块简介

 

1.运行时数据区:经过编译生成的字节码文件(class文件),由 class loader(类加载子系统)加载后交给执行引擎执行。在执行引擎执行的过程中产生的数据会存储在一块内存区域。这块内存区域就是运行时区域

 

2.程序计数器(线程私有:用于记录当前线程的正在执行的字节码指令位置。由于虚拟机的多线程是切换线程并分配 cpu 执行时间的方式实现的,不同线程的执行位置都需要记录下来,因此程序计数器是线程私有的。

  • 多个线程竞争时被挂起,程序计数器记录执行到哪里
  • 唯一一个在 Java 虚拟机规范中没有规定任何 OutOfMemoryError 情况的区域

 

3.虚拟机栈(线程私有:虚拟机栈是 Java 方法执行的内存结构,虚拟机会在每个 Java 方法执行时创建一个“栈桢”,用于存储局部变量表,操作数栈,动态链接,方法出口等信息。当方法执行完毕时,该栈桢会从虚拟机栈中出栈。其中局部变量表包含基本数据类型和对象引用

在 Java 虚拟机规范中,对这个区域规定了两种异常状态:如果线程请求的栈的深度大于虚拟机允许的深度,将抛出 StackOverFlowError 异常(栈溢出);如果虚拟机栈可以动态扩展(现在大部分 Java 虚拟机都可以动态扩展,只不过 Java 虚拟机规范中也允许固定长度的 Java 虚拟机栈),如果扩展时无法申请到足够的内存空间,就会抛出 OutOfmMemoryError 异常(没有足够的内存)

 

4.本地方法栈(线程私有:java调用非java代码的接口,用于与java环境外交互、与操作系统交互以及Sun's Java。

Java 方法的执行有虚拟机栈,本地方法(native关键字的执行则对应有本地方法栈。

本地方法一般是用其它语言(C、C++ 或汇编语言等)编写的,并且被编译为基于本机硬件和操作系统的程序。

 

5.方法区(线程共享:用于存储已被虚拟机加载的信息,常量,静态变量,即时编译器编译后的代码等数据。

该区域的垃圾回收——主要目标是对常量池的回收和对类的卸载,但是一般比较难实现。

 

6.Java 堆(Heap):堆的主要作用是存放程序运行过程中创建的对象实例,因为要存放的对象实例有可能会极多,因此也是虚拟机内存管理中最大的一块。并且由于硬件条件有限,所以需要不断回收已“无用”的实例对象来腾出空间给新生成的实例对象。因此 Java 的垃圾回收主要是针对堆进行回收的(还有方法区的常量池),Java 堆被称为GC堆(Garbage Collected Heap)。

Java 堆不需要连续内存,并且可以动态增加其内存,增加失败会抛出 OutOfMemoryError 异常。

 

现代的垃圾收集器基本都是采用分代收集算法(因为对象的生命周期不一样),主要思想是针对不同的对象采取不同的垃圾回收算法。虚拟机把 Java 堆分成以下三块:新生代、老年代、永久代。

而新生代在三个区域中垃圾回收的频率最高。

为了更高效地进行垃圾回收,把新生代继续划分成以下三个空间:Eden(伊甸园)、From Survivor(幸存者)、To Survivor

 

7.类加载机制(Class Loader):类加载子系统是根据一个类的全限定名来加载该类的二进制流到内存中,在JVM中将形成一份描述 Class 结构的元信息对象(方法区),通过该元信息对象可以获知 Class 的结构信息:如构造函数,属性和方法等,Java 允许用户借由这个 Class 相关的元信息对象间接调用 Class 对象的功能。

 

JVM如何工作?

如上面的体系结构图所示,JVM 分为三个主要的子系统:

1.类加载器子系统

Java的动态类加载功能由类加载器子系统处理。它负责加载链接,并且在运行时首次引用类的时候初始化类,而非编译期间。

2.运行时数据区

3.执行引擎

分配到运行时数据区的字节码将被执行引擎执行。执行引擎读取字节码并逐一执行。

  • 解释器 —— 解释器能更加快速地解释字节码,但是执行缓慢。解释器的缺点是当多次调用一个方法时,每次都要重新解释。
  • JIT编译器 —— JIT编译器弥补了解释器的不足。执行引擎使用解释器来转换字节码,当它发现重复的代码时,它将使用JIT编译器来编译整个字节码并转换为本地代码。本地代码将直接被重复的方法所调用,从而提高系统性能。
  • 中间代码生成器 —— 生成中间代码。
  • 代码优化器 —— 负责优化上述生成的中间代码。
  • 目标代码生成器 —— 负责生成机器码或者本地代码。
  • 分析器 —— 一个特殊的组件,负责查找热点代码,比如一个方法是否被调用多次。
  • 垃圾回收器 —— 回收并删除未引用的对象。可以通过调用System.gc()来触发垃圾回收,但不能保证它执行。注意:JVM的垃圾回收是回收被创建的对象。

Java本地接口(JNI)JNI本地方法库交互,并为执行引擎提供本地方法库

本地方法库(Native Method Libraries):它是执行引擎所需的本地库集合。

 

未完待续

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值