【Java面试】JVM

文章目录

一、JVM内存结构

1、Jdk和Jre和JVM的区别?

JDK包括了JRE,JRE包括了JVM。
JDK是我们编写代码使用的开发工具包,JRE是Java运行时环境,JVM俗称Java虚拟机,是Java运行环境的一部分。


2、说一下JVM由哪些部分组成,运行流程是什么?

JVM包含两个子系统和两个组件:两个子系统为Class loader(类加载器)、Execution engine(执行引擎);两个组件为Runtime data area(运行时数据区)、Native Interface(本地接口)。如图:
在这里插入图片描述

Class loader(类加载器):根据给定的全限定类名(如:java.lang.String)来加载class文件到Runtime data area中的method area。
Execution engine(执行引擎):执行classes中的指令。
Native Interface(本地接口):与本地native libraries交互,是底层C++语言交互的接口。
Runtime data area(运行时数据区域):这就是我们常说的JVM内存。

JVM运行流程:
首先通过编译器把 Java 代码转换成字节码,类加载器(Class Loader)再把字节码加载到内存中,将其放在运行时数据区(Runtime data area)的方法区内,而字节码文件只是 JVM 的一套指令集规范,并不能直接交给底层操作系统去执行,因此需要特定的命令解析器执行引擎 (Execution Engine),将字节码翻译成底层系统指令,再交由 CPU 去执行,而这个过程中需要调用其他语言的本地库接口(Native Interface)来实现整个程序的功能。


3、说一下JVM运行时数据区

Java虚拟机所管理的内存(运行时数据区)被划分为如下几个区域:
在这里插入图片描述

(1)程序计数器(Program Counter Register)
线程私有,当前线程所执行字节码的行号执行器,字节码解析器的工作是通过改变这个计数器的值,来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能,都需要依赖这个计数器来完成。

为什么需要线程计数器?因为线程是不具备记忆功能。

(2)Java虚拟机栈(Java Virtual Machine Stacks)
线程私有,每个方法在执行的同时都会在Java虚拟机栈中创建一个栈帧,用于存储局部变量表、操作数栈、动态链接、方法出口等信息。

(3)本地方法栈(Native Method Stack)
与虚拟机栈的作用是一样的,只不过虚拟机栈是服务 Java 方法的,而本地方法栈是为虚拟机调用 Native 方法服务的。

(4)Java 堆(Java Heap)
Java 虚拟机中内存最大的一块,是被所有线程共享的,几乎所有的对象实例都在这里分配内存。

(5)方法区(Methed Area)
用于存储已被虚拟机加载的类信息、常量、静态变量、编译后的字节码等数据。


4、详细的介绍下程序计数器?(重点理解)

(1)程序计数器是一块较小的内存空间,线程私有的,它可以看作是:保存当前线程所正在执行的字节码指令的地址 (行号)

(2)由于Java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,一个处理器 都只会执行一条线程中的指令。因此,为了线程切换后能恢复到正确的执行位置,每条线程都有一个独立的程序计数器,各个线程之间计数器互不影响,独立存储。称之为“线程私有”的内存。程序计数器内存区域是虚拟机中唯一没有规定OutOfMemoryError情况的区域


5、详细介绍下Java虚拟机栈?(重点理解)

(1)Java虚拟机是线程私有的,它的生命周期和线程相同。
(2)虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。

解析栈帧:
(1)局部变量表:用来存储临时8个基本数据类型、对象引用地址、返回值类型数据。
(2)操作数栈:例如代码中有个 i = 6*6,它在一开始就会进行操作,读取代码时,进行计算后再放入局部变量表中去。
(3)动态链接:假如方法中,有个service.add()方法,要链接到别的方法中去,这就是动态链接,用来存储链接信息。
(4)方法出口:出口正常的话就是return,不正常的话就是抛出异常。


6、详细介绍下Java堆?(重点理解)

(1)Java堆(Java Heap)是Java虚拟机所管理的内存中最大的一块,是被所有线程共享的一块内存区域,在虚拟机启动时创建,此内存区域目的就是存放对象实例
(2)在Java虚拟机规范中描述的是:所有对象实例以及数组都要在堆上分配。
(3)Java堆是垃圾回收管理器的主要区域,因此也被称为“GC堆”。
(4)从内存回收角度来看Java堆可分为:新生代和老年代。
(5)从内存分配的角度看,线程共享的Java堆中可能划分出多个线程私有的分配缓冲区。
(6)根据Java虚拟机规范规定,Java堆可以处于物理上不连续的内存空间中。当前主流虚拟机都是可扩展的(通过-Xmx和-Xms控制)。如果堆中不能再完成对象实例分配,并且也无法再扩展时,将会抛出OutOfMemoryError异常。


7、解释一下本地方法栈?

(1)本地方法栈,和方法栈类似,也是属于线程私有的,只不过是native方法产生的栈为本地方法栈。
(2)native方法体是看不到的,必须要去Oracle官网去下载才能看得到,而且native关键字修饰的大部分源码都是C和C++的代码。


8、详细介绍一下方法区?(重点理解)

方法区是所有线程共享的内存区域,它用于存储已被Java虚拟机加载的类模板信息、常量、静态变量、编译后的字节码等数据。

它有个别名叫Non-Heap(非堆)。当方法区无法满足内存分配时,也会抛出OutOfMemoryError。


9、什么是JVM字节码执行引擎?

虚拟机核心的组件就是执行引擎,它负责执行虚拟机的字节码,将字节码解析成操作系统能识别的指令后,才能在系统上执行。


10、你听说过直接内存吗?

直接内存并不是虚拟机运行时数据区中的,也不是Java虚拟机中定义的内存区域。但是这部分内存也被频繁地使用,而且也可能导致OutOfMemoryError出现。

我的理解就是直接内存是基于物理内存和Java虚拟机内存的中间内存


11、堆栈的区别(重点)

JVM堆 JVM栈
物理地址 堆中对象的物理地址分配是不连续的,因此性能慢些。在GC的时候也要考虑不连续的分配,所以有各种算法。比如:标记清除,复制,标记压缩,分代(新生代使用复制算法,老年代使用标记压缩)。 栈使用的数据结构中的栈,先进后出原则,物理地址分配是连续的,所以性能快。
内存方面 堆因为是不连续的,所以分配的内存是在运行期确定的,因此大小不固定,一般堆大小远远大于栈 栈是连续的,所以分配的内存大小在编译期就确认,大小是固定的。
存放内容 堆存放的是对象的实例和数组,该区更关注的是数据的存储。 栈存放:局部变量表,操作数栈,动态链接,返回数据等。栈关注的是程序方法的执行。</
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值