面试—JVM

2024-09-07 09:20
文章标签 java jvm 面试

本文主要是介绍面试—JVM,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

JVM内存结构

类的生命周期

双亲委派机制

打破双亲委派机制

垃圾回收机制

判断垃圾回收算法

垃圾回收算法

G1垃圾回收器


JVM内存结构

  1. 程序计数器

    记录要执行的字节码指令的地址,可以控制程序指令的进行,实现分支、跳转、异常等

    在多线程执行时,会记录线程切换前执行到的那一条指令

  2. Java虚拟机栈

    每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、帧数据(动态连接、方法出口、异常表的引用)的信息

  3. 本地方法栈

    与java虚拟机栈作用相似

    区别 java虚拟机栈存储Java方法(字节码)调用时的栈帧

    本地方法栈存储的是native本地方法的栈帧

  4. 内存中空间最大的,创建的对象都存储在堆内存上

    三个值used,total,max,由Java虚拟机分配内存,一般设置total=max

  5. 方法区

    存放基础信息的位置,线程安全,主要由三个部分

    1. 类的元信息

      存放每个类的基本信息,一般为instanceKlass对象,在类的加载阶段完成

      主要是字节码文件的基本信息:包括魔数、版本号、父类、接口、常量池、字段、方法、虚方法表等信息

    2. 运行时常量池

      主要是类的加载和连接阶段,将符号引用编译为地址引用,通过地址快速定位到常量池中的内容

    3. 字符串常量池

      主要存放常量字符串(JDK7之后将字符串常量池放入到堆内存中)

类的生命周期

  • 主要分为五个阶段,加载—>连接—>初始化—>使用—>卸载

  • 在连接阶段又分为三个部分,验证—>准备—>解析

  1. 加载

    1. 类加载器根据类的全限定类名以二进制流的方式获取字节码文件

    2. 加载完之后,java虚拟机将字节码的信息保存在内存的方法区中

    3. 同时java虚拟机还会在堆中生成一份与方法区中数据类似的java.lang.Class对象(开发者主要的访问,可以很好的控制开发者访问数据的范围)

  2. 连接

    1. 验证

      验证是否为字节码文件,验证魔术(0XCACFBABE),版本号等信息

    2. 准备

      给静态变量赋值

    3. 解析

      将符号引用解析为内存地址引用

  3. 初始化

    执行静态代码块的内容,并为静态变量赋值

  4. 使用

    类的创建,赋值,调用等

  5. 卸载

    垃圾回收

双亲委派机制

  • 类加载器分为启动类加载器、扩展类加载器、应用程序类加载器和自定以类加载器

  • 每一个加载器中都有一个parent的常量,用于指向父类加载器

  1. 应用程序类加载器的父类加载器是扩展类加载器

  2. 自定义类加载器—>应用程序类加载器—>扩展类加载器—>启动类加载器

双亲委派机制

  1. 当一个类被加载的时候,会自下向上进行查找,委派父类加载器去尝试加载,如果启动类加载器收到请求能够找到并加载这个类,则加载过程结束,返回class对象

  2. 如果启动类加载器不能进行加载,就会向下委派子类加载器进行加载

  • 优点

    • 避免恶意代码替换JDK核心类库,如java.lang.String

    • 避免类被重复加载

打破双亲委派机制

  • ClassLoder主要负责将类的字节码加载到 JVM(Java 虚拟机)中,并创建对应的Class对象。

  • 核心原理

    • loadClass() 类加载器的入口,内部调用findClass()方法

    • findClass() 根据类的全限定类名,找到字节码文件

    • defineClass() 做一些类的校验,然后调用虚拟机底层方法将字节码信息加载到内存中

    • resolveClass() 执行类生命周期的连接阶段(验证、准备、解析)

  1. 自定义类加载器打破

    自定义类继承ClassLoder,重写loadClass()方法

    1. 传递类的全限定类名,找到字节码文件,并加载到内存中,变成二进制的数组

    2. 调用defineClass将二进制的数组传递,在方法区和堆中生成对应的数据

  2. 线程上下文类的加载器(JDBC)

    启动类加载器加载DriverManager,又需要通过SPI机制拿到应用程序类加载器去加载jar包中的mysql驱动(由启动类加载器委派应用程序类加载器去加载类,打破了双亲委派机制)

    SPI机制

    1. 在驱动的jar包中暴露出需要被加载的类,放到固定的文件中

    2. DriverManager中使用ServiceLoader方法加载文件中的类名,并且使用类加载器加载对应的类并创建对象

    • 由于启动类加载器在加载完DriverManager完之后,通过初始化阶段触发了应用程序类加载器的加载,类的加载过程依然符合双亲委派机制,只是触发了SPI机制,所以没有打破双亲委派机制

  3. OSGI模块化

    在OSGI中,每个模块都有自己独立的加载器,当一个模块的类加载器收到加载请求时,会先在自己模块进行查找,找不到回去别的模块进行查找。相当于同级之间的类加载器委派加载

垃圾回收机制

判断垃圾回收算法

  1. 引用计数法

    在堆内存中,对象被引用一次,就会计数器+1。如果计数器为0,则进行回收

    若时堆内存的对象互相引用,则无法回收,会造成内存溢出

  2. 可达性分析算法

    将对象分为垃圾回收的根对象(GCRoot对象)和普通对象

    与GCRoot对象形成一个引用链

    1. 强引用,引用的对象不会被回收

    2. 软引用,SoftReference对象(与GCRoot对象是强引用关联)来实现软引用,当对象使用完之后不会马上进行垃圾回收,而是当内存不足的时候进行垃圾回收,回收软引用的对象

    3. 弱引用,与软引用相似,但是弱引用不论内存是否不足都会被直接回收

    4. 虚引用

    5. 终结引用

垃圾回收算法

  1. 标记—清除算法

    使用可达性分析算法,标记出存活的对象(与GcRoot强引用的对象),删除内存中的非存活对象

  2. 复制算法

    把内存分为两个空间From(对象分配的空间)和To

    1. GC阶段,先将GCRoot对象搬运到To空间

    2. 将与GCRoot对象关联的对象也搬运到To空间

    3. 将两块空间的名字互换

    4. 清理From空间

  3. 标记—整理算法

    由于标记清楚算法清理后的内存时碎片化的,分配时内存空间不合适

    先标记存活对象,将存活的对象移动到堆的一端,清理非存活的对象

  4. 分代回收算法

    分为新生代(Young区)和老年代(Old区)

    新生代又分为伊甸园区(Eden区)和幸存者区(Survivor)

    1. 伊甸园区主要是存放新创建的对象

    2. 当伊甸园区存满时,会触发新生代的垃圾回收

      幸存者区主要使用复制算法进行垃圾回收

    3. 每次新生代垃圾回收一次会为对象记录年龄,当年龄到达阀值(最大15),对象就会晋升为老年代

G1垃圾回收器

  • 主要是分代回收算法,但是将内存分为大小相同的区域(Region)大小是堆空间/2048

  1. 当G1判断年轻代内存不足60%时,会触发新生代的垃圾回收

  2. 幸存者区存活的对象会被搬运的另一个幸存者区

  3. 当对象年龄到达15,放到老年代中

  4. 部分对象的大小如果超过区域的一半时,会直接放入单独的老年代中(Humongous区)

  5. 多次回收后,出现多个老年代,达到阀值时(默认45%),会触发混合回收,将新生代与老年代的对象使用复制算法回收

这篇关于面试—JVM的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1144677

相关文章

springboot的调度服务与异步服务使用详解

《springboot的调度服务与异步服务使用详解》本文主要介绍了Java的ScheduledExecutorService接口和SpringBoot中如何使用调度线程池,包括核心参数、创建方式、自定... 目录1.调度服务1.1.JDK之ScheduledExecutorService1.2.spring

将java程序打包成可执行文件的实现方式

《将java程序打包成可执行文件的实现方式》本文介绍了将Java程序打包成可执行文件的三种方法:手动打包(将编译后的代码及JRE运行环境一起打包),使用第三方打包工具(如Launch4j)和JDK自带... 目录1.问题提出2.如何将Java程序打包成可执行文件2.1将编译后的代码及jre运行环境一起打包2

Java使用Tesseract-OCR实战教程

《Java使用Tesseract-OCR实战教程》本文介绍了如何在Java中使用Tesseract-OCR进行文本提取,包括Tesseract-OCR的安装、中文训练库的配置、依赖库的引入以及具体的代... 目录Java使用Tesseract-OCRTesseract-OCR安装配置中文训练库引入依赖代码实

Java中对象的创建和销毁过程详析

《Java中对象的创建和销毁过程详析》:本文主要介绍Java中对象的创建和销毁过程,对象的创建过程包括类加载检查、内存分配、初始化零值内存、设置对象头和执行init方法,对象的销毁过程由垃圾回收机... 目录前言对象的创建过程1. 类加载检查2China编程. 分配内存3. 初始化零值4. 设置对象头5. 执行

SpringBoot整合easy-es的详细过程

《SpringBoot整合easy-es的详细过程》本文介绍了EasyES,一个基于Elasticsearch的ORM框架,旨在简化开发流程并提高效率,EasyES支持SpringBoot框架,并提供... 目录一、easy-es简介二、实现基于Spring Boot框架的应用程序代码1.添加相关依赖2.添

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程

《SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程》本文详细介绍了如何在虚拟机和宝塔面板中安装RabbitMQ,并使用Java代码实现消息的发送和接收,通过异步通讯,可以优化... 目录一、RabbitMQ安装二、启动RabbitMQ三、javascript编写Java代码1、引入

spring-boot-starter-thymeleaf加载外部html文件方式

《spring-boot-starter-thymeleaf加载外部html文件方式》本文介绍了在SpringMVC中使用Thymeleaf模板引擎加载外部HTML文件的方法,以及在SpringBoo... 目录1.Thymeleaf介绍2.springboot使用thymeleaf2.1.引入spring

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满