0. 前言

Java虚拟机(Java virtualmachine)实现了Java语言最重要的特征:即平台无关性。

平台无关性原理:编译后的 Java程序(.class文件)由 JVM执行。JVM屏蔽了与具体平台相关的信息,使程序可以在多种平台上不加修改地运行。Java虚拟机在执行字节码时,把字节码解释成具体平台上的机器指令执行。因此实现Java平台无关性。

1. JVM结构图

JVM = 类加载器 classloader + 执行引擎 executionengine + 运行时数据区域 runtime data area

首先Java源代码文件被Java编译器编译为字节码文件,然后JVM中的类加载器加载完毕之后,交由JVM执行引擎执行。在整个程序执行过程中,JVM中的运行时数据区(内存)会用来存储程序执行期间需要用到的数据和相关信息。

因此,在Java中我们常常说到的内存管理就是针对这段空间进行管理(如何分配和回收内存空间)。

2. ClassLoader

Classloader把硬盘上的class文件加载到JVM中的运行时数据区域,但是它不负责这个类文件能否执行,而这个是执行引擎负责的。

限于篇幅,类加载器的组织结构,加载类的机制原理等会在JVM——类加载器总结一文中描述。

双亲委派模型以及自定义类加载器会在JVM——自定义类加载器一文中描述。

3. 执行引擎

作用:执行字节码,或者执行本地方法。

4. Runtime DataArea

JVM在运行期间,在运行时数据区对JVM内存空间的划分和分配,划分为了以下几个区域来存储。

(图注:JDK1.7已经把常量池转移到堆里面了!)

  • PC计数器(The PC Register)
  • (1)每一个Java线程都有一个PC寄存器,用以记录比如在线程切换回来后恢复到正确的执行位置。

  • (2)如该线程正在执行一个Java方法,则计数器记录的是正在执行的虚拟机字节码地址,如执行native方法,则计数器值为空。

  • (3)此内存区域是唯一一个在JVM中没有规定任何OutOfMemoryError情况的区域。

  • JVM栈(Java Virtual MachineStacks)
  • (1)JVM栈是线程私有的,并且生命周期与线程相同。并且当线程运行完毕后,相应内存也就被自动回收。

  • (2)栈里面存放的元素叫栈帧,每个方法从调用到执行结束,其实是对应一个栈帧的入栈和出栈。栈帧用于存储执行方法时的一些数据,如局部变量表、操作数栈(执行引擎计算时需要),方法出口等等。

  • (3)这个区域可能有两种异常:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常(如:将一个函数反复递归自己,最终会出现这种异常)。如果JVM栈可以动态扩展(大部分JVM是可以的),当扩展时无法申请到足够内存则抛出OutOfMemoryError异常。

  • 本地方法栈(Native Method Stacks)
  • (1)本地方法栈与JVM栈的作用很相似,他们的区别在于虚拟机栈是为执行Java代码方法服务,而本地方法栈是为Native方法服务。

  • (2)和JVM栈一样,这个区域也会抛出StackOverflowError和OutOfMemoryError异常。

  • 方法区(Method Area)
  • (1)方法区域是全局共享的,比如每个线程都可以访问同一个类的静态变量。在方法区中,存储了已被JVM加载的类的信息、静态变量、编译器编译后的代码等。如,当程序中通过getName、isInterface等方法来获取信息时,这些数据来源于方法区。

  • (2)由于使用反射机制的原因,虚拟机很难推测哪个类信息不再使用,因此这块区域的回收很难!另外,对这块区域主要是针对常量池回收,值得注意的是JDK1.7已经把常量池转移到堆里面了。

  • (3)同样,当方法区无法满足内存分配需求时,会抛出OutOfMemoryError。

  • 运行时常量池(Runtime Constant Pool)
  • (1)存放类中固定的常量信息、方法引用信息等,其空间从方法区域(JDK1.7后为堆空间)中分配。

  • (2)Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有就是常量表,用于存放编译期已可知的常量,这部分内容将在类加载后进入方法区(永久代)存放。但是Java语言并不要求常量一定只有编译期预置入Class的常量表的内容才能进入方法区常量池,运行期间也可将新内容放入常量池(最典型的String.intern()方法)。

  • (3)当常量池无法在申请到内存时会抛出OutOfMemoryError异常,上面也分析过了。

  • Java堆
  • (1)Java堆是JVM所管理的最大的一块内存。它是被所有线程共享的一块内存区域,在虚拟机启动时创建。

  • (2)几乎所有的实例对象都是在这块区域中存放。(JIT编译器貌似不是这样的)。

  • (3)Java堆是垃圾收集管理的主要战场。所有Java堆可以细分为:新生代和老年代。再细致分就是把新生代分为:Eden空间、FromSurvivor空间、To
    Survivor空间。JVM具体的垃圾回收机制总结请查看我的另外一篇JVM——内存管理和垃圾回收。

  • (4)根据Java虚拟机规范的规定,Java堆可以处于物理上不连续的内存空间中,只要逻辑上是连续的即可。

如果在堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出OutOfMemoryError异常。

5. 堆和栈的区别

这是一个非常常见的面试题,主要从以下几个方面来回答。

  • (1)各司其职

最主要的区别就是栈内存用来存储局部变量和方法调用信息。
而堆内存用来存储Java中的对象。无论是成员变量、局部变量还是类变量,它们指向的对象都存储在堆内存中。

  • (2)空间大小

栈的内存要远远小于堆内存,如果你使用递归的话,那么你的栈很快就会充满并产生StackOverFlowError。
关于如何设置堆栈内存的大小,可以查看JVM——内存管理和垃圾回收中的相关介绍。

  • (3)独有还是共享

栈内存归属于线程的私有内存,每个线程都会有一个栈内存,其存储的变量只能在其所属线程中可见。
而堆内存中的对象对所有线程可见,可以被所有线程访问。

  • (4)异常错误

如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常。

如果JVM栈可以动态扩展(大部分JVM是可以的),当扩展时无法申请到足够内存则抛出OutOfMemoryError异常。

而堆内存没有可用的空间存储生成的对象,JVM会抛出java.lang.OutOfMemoryError。

本文转自

JVM统介——Java虚拟机架构相关推荐

  1. JVM——Java虚拟机架构

    Java虚拟机(Java virtualmachine)实现了Java语言最重要的特征:即平台无关性. 平台无关性原理:编译后的 Java程序(.class文件)由 JVM执行.JVM屏蔽了与具体平台 ...

  2. 【JVM】<Java虚拟机>JVM架构各种**虚拟机

    目录 一.Java代码执行流程: 二.JVM架构模型: 1.这两种架构之间的区别: 2.反编译指令: 在IDEA中查看字节码: 三.JVM的生命周期: 1.虚拟机的启动: 2.虚拟机的执行: 3.虚拟 ...

  3. JVM内幕:Java虚拟机详解

    这篇文章解释了Java 虚拟机(JVM)的内部架构.下图显示了遵守 Java SE 7 规范的典型的 JVM 核心内部组件. 上图显示的组件分两个章节解释.第一章讨论针对每个线程创建的组件,第二章节讨 ...

  4. [开源JVM] yvm - 自制Java虚拟机

    中文 | English | | | YVM是用C++写的一个Java虚拟机,现在支持Java大部分功能,以及一个基于标记清除算法的并发垃圾回收器. 不过还有很多bug等待修复. 感兴趣的朋友pull ...

  5. JVM笔记:Java虚拟机的字节码指令详解

    1.字节码 Java能发展到现在,其"一次编译,多处运行"的功能功不可没,这里最主要的功劳就是JVM和字节码了,在不同平台和操作系统上根据JVM规范的定制JVM可以运行相同字节码( ...

  6. JVM II(Java虚拟机,Java Virtual Machine)

    文章目录 7.助记符 相关助记符 8.JVM的4种引用级别 (1)强引用 (2)软引用(SoftReference) 软引用被GC回收例子: (3)弱引用(WeakReference) (4)虚引用( ...

  7. jvm十五:java虚拟机内存图

  8. jvm 调优 java 虚拟机 马士兵 马士兵 马士兵 笔记

    文档连接:http://download.csdn.net/download/disalong/10259953 例子代码连接:http://download.csdn.net/download/di ...

  9. Java虚拟机(JVM)简介

    什么是JVM Java虚拟机(JVM)是使计算机能够运行Java程序的抽象计算机. JVM有三个概念: 1.规格 2.实施 3.实例. 该规范是正式描述JVM实现要求的文档. 具有单一规范可确保所有实 ...

最新文章

  1. Oracle Schema Objects——Index
  2. 2011年3月华章新书书讯:ASP.NET本质论、Erlang编程指南、SNS网站构建
  3. shell调用python函数_shell调用python函数
  4. python 面试题(1)--- python模块
  5. 2010.03.13 微软VSTS2008 动手实验室
  6. 试题 15 枚举n-m之间的所有质数(质数)
  7. Iframe相关操作
  8. 2020年最新世界地图_2020年标准地图发布,为何要发布新版标准地图?
  9. 解决lenovo V470 安装win8 无法重启、关机故障
  10. 使用Windows Live Writer WLW向Joomla网站发帖
  11. 2021年暑期训练阶段三Day3
  12. 微博html5版开视频怎么退出,微博怎么取消视频号?微博视频号怎么关闭
  13. 6.2.1 字符串方法 upper()、 lower()、 isupper()和 islower()
  14. 优酷 Android 包瘦身治理思路全解
  15. windows的ping测试脚本
  16. Termux外置硬盘挂载——rclone WebDav 挂载网盘
  17. segger j-flash使用笔记
  18. C语言中getchar()函数的详解
  19. 4个基本不等式的公式高中_基本不等式公式四个叫什么名字
  20. 去雾模块dehaze.py(可直接调用)

热门文章

  1. 70. Climbing Stairs
  2. 使用 json_serializable (flutter packages pub run build_runner build) 问题
  3. 浅析软件工程中的UML建模技术
  4. LeetCode119.杨辉三角II
  5. JAVA方法调用中的解析与分派
  6. 【ACM】最长公共子序列 - 动态规划
  7. C++ code:数值计算之矩形法求解积分问题
  8. DropDownList设置选定项,设置选择项,最安全的方法
  9. 光脚丫学LINQ(025):如何验证DBML和外部映射文件
  10. 基于fiddler的网络爬虫校园网自动登陆系统