秋招后端开发面试题 - JVM底层原理


目录

  • JVM底层原理
    • 前言
    • 面试题
      • Java 对象的创建过程?
      • 什么是指针碰撞?什么是空闲列表?/ 内存分配的两种方式?
      • JVM 里 new 对象时,堆会发生抢占吗?JVM 是怎么设计来保证线程安全的?/ 内存分配并发问题?
      • 对象的内存布局?
      • 对象怎么访问定位?
      • 内存溢出和内存泄漏?
      • 能手写内存溢出的例子吗?
      • 内存泄漏可能由哪些原因导致呢?
      • 如何判断对象仍然存活?/ 如何判断对象是否死亡?
      • Java 中可作为 GC Roots 的对象有哪几种?
      • 说一下对象有哪几种引用?/ 强引用、软引用、弱引用、虚引用?
      • 什么是 Stop The World?
      • 什么是 OopMap?
      • 对象一定分配在堆中吗?有没有了解逃逸分析技术?
      • 线上服务 CPU 占用过高怎么排查?
      • 内存飙高问题怎么排查?
      • 举例栈溢出的情况?
      • 调整栈大小,就能保证不出现溢出吗?
      • 分配的栈内存越大越好吗?
      • 垃圾回收是否会涉及到虚拟机栈
      • 方法中定义的局部变量是否线程安全?
      • 静态变量和局部变量?
      • 如何判断一个常量是废弃常量?


JVM底层原理

前言

已经找到工作了,分享秋招时的笔记。祝大家都能顺利找到自己心仪的工作。


面试题

Java 对象的创建过程?

  1. 类加载检查:
    • 当虚拟机遇到 new 指令时,首先检查这个指令的参数,也就是要创建的对象的类是否已经加载过
    • 如果没有加载过,虚拟机会执行类加载过程
  2. 分配内存:
    • 类加载检查通过后,虚拟机会为新对象分配内存空间
  3. 对象内存初始化:
    • 分配内存完成后,虚拟机会将分配到的内存空间初始化为零值
  4. 设置对象头:
    • 虚拟机会在对象的内存空间中设置对象头,用于存储关于对象的元数据信息
  5. 执行初始化方法(构造函数):
    • 经过上述步骤,一个新的对象已经产生,但是从 Java 程序的角度来看,对象的创建还没有完成
    • 然后,会根据程序员定义的构造函数进行初始化

什么是指针碰撞?什么是空闲列表?/ 内存分配的两种方式?

指针碰撞:

  • 指针碰撞假定 Java 堆中的内存是绝对规整的
  • 内存的分界点由一个指针作为指示器来标示,指向已分配内存的末尾
  • 在分配对象内存时,只需要将指针向空闲空间方向移动对象内存大小的位置
  • 适用于基于压缩策略的收集器,例如 Serial 和 ParNew 收集器

空闲列表:

  • 空闲列表假设 Java 堆内存并不规整,已分配内存和空闲内存交错分布
  • 虚拟机维护一个空闲列表,记录哪些内存块是可用的,即未被分配的
  • 在分配对象内存时,虚拟机会在空闲列表中找到一块足够大的空间来分配给对象。分配后,虚拟机需要更新空闲列表上的记录,标记分配的区域为已用
  • 适用于基于清除算法的收集器,例如 CMS 收集器

JVM 里 new 对象时,堆会发生抢占吗?JVM 是怎么设计来保证线程安全的?/ 内存分配并发问题?

  • CAS:使用 CAS 操作来保证更新操作的原子性
  • 本地线程分配缓冲 (TLAB):
    每个线程在 Java 堆中预先分配一小块内存,称为本地线程分配缓冲
    要分配内存的线程,先在本地缓冲区中分配,只有本地缓冲区用完,分配新的缓冲区才需要同步锁定

对象的内存布局?

对象在堆内存中的布局可以划分为三个部分:对象头、实例数据和对齐填充

  • 对象头:包括两部分信息:存储对象自身的运行时数据;类型指针
  • 实例数据:用来存储对象真正的有效信息
  • 对齐填充:起占位符的作用

对象怎么访问定位?

使用句柄:

  • Java 堆中划分内存来作为句柄池
  • 句柄中包含对象实例数据的指针和对象类型数据的指针
    image-20230317153832919

直接指针:

  • 对象的实例数据直接存放在堆内存中
    image-20230317154017974

对比:

  • 使用句柄:在对象被移动的时候,不需要更新引用地址
  • 直接指针:效率高,节省了一次指针定位的时间开销(HotSpot 使用)

内存溢出和内存泄漏?

  • 内存溢出:申请的内存超过可用内存,内存不足
  • 内存泄漏:申请的内存空间没有被正确释放,导致内存空间被浪费

能手写内存溢出的例子吗?

  • Java 堆溢出:Java 堆用于存储对象实例,只要不断创建不可回收的对象,比如静态变量,随着对象数量的增加,总容量超过最大堆的限制就会产生 OOM
    public static void main(String[] args) {List<OOMObject> list = new ArrayList<>();while (true) {list.add(new OOMObject());}}
  • 虚拟机栈溢出:不停创建线程,也会出现 OOM 异常
    public static void recursiveMethod() {recursiveMethod();}public static void main(String[] args) {try {recursiveMethod();} catch (Throwable e) {System.out.println("Stack depth: " + e.getStackTrace().length);e.printStackTrace();}}

内存泄漏可能由哪些原因导致呢?

  • 静态集合: 静态集合生命周期与 JVM 相同,如果将对象添加到静态集合中并且忘记删除,这些对象将一直存在于内存中,无法被垃圾回收
  • 单例模式: 单例模式中的实例会被以静态变量的方式存储在内存中,一旦创建,会在整个 JVM 生命周期中存在。如果单例对象占用内存过多或者被误用,就可能导致内存泄漏
  • 连接未释放: 如果在使用完数据库连接、网络连接等资源后没有正确关闭,这些资源可能不会被释放,导致内存泄漏
  • 变量作用域过大: 如果变量的作用域超出了实际需要的范围,导致对象不能被及时释放,就会引起内存泄漏
  • hash 值发生改变: 如果对象的 hashCode 值在存入哈希容器后被修改,那么在尝试从哈希容器中获取该对象时,哈希容器会根据 hashCode 去查找,但实际上已经找不到这个对象了
  • ThreadLocal 使用不当: ThreadLocal 中的 key 是弱引用,但 value 是强引用。如果 ThreadLocal 的使用不当,导致 key 无法被垃圾回收,而 value 却一直存在,就会造成内存泄漏

如何判断对象仍然存活?/ 如何判断对象是否死亡?

引用计数算法:

  • 在对象中添加一个引用计数器
  • 每当有一个地方引用它时,计数器值就 +1
  • 引用失效时,计数器值 -1
  • 任何时刻计数器为零的对象就是不可能再被使用的
  • 不使用该方法,因为很难解决对象之间相互循环引用的问题

可达性分析算法:

  • 通过一系列 GC Roots 的对象作为起点
  • 从这些节点开始向下搜索,节点走过的路径就是引用链
  • 当一个对象到 GC Roots 没有任何引用链相连,证明此对象是不可能再被使用的
    image-20230317160832041

Java 中可作为 GC Roots 的对象有哪几种?

  • 虚拟机栈中引用的对象
  • 本地方法栈中引用的对象
  • 方法区中类静态属性引用的对象
  • 方法区中常量引用的对象

说一下对象有哪几种引用?/ 强引用、软引用、弱引用、虚引用?

强引用:

  • 使用最普遍的引用
  • 无论任何情况,只要具有强引用,垃圾收集器就永远不会回收被引用的对象

软引用:

  • 用来描述还有用,但非必须的对象
  • 只被软引用关联着的对象,在系统将要发生内存溢出异常前,会把这些对象列进回收范围之中进行第二次回收

弱引用:

  • 用来描述那些非必须的对象,强度比软引用还弱一些
  • 被弱引用关联的对象只能生存到下一次垃圾收集发生为止
  • 无论当前内容是否足够,都会回收掉只被弱引用关联的对象

虚引用:

  • 不会决定对象的生命周期
  • 唯一目的是为了能在这个对象被收集器回收时收到一个系统通知

什么是 Stop The World?

  • 在垃圾回收的过程中,会涉及到对象的移动。为了保证对象引用更新的正确性,必须暂停所有的用户线程,即 Stop The World

什么是 OopMap?

  • 类加载完成后,记录对象偏移量和数据类型的映射表

对象一定分配在堆中吗?有没有了解逃逸分析技术?

  • 对象不一定分配在堆中
  • 逃逸分析是一种编译器技术,用于确定对象创建后从方法逃逸到哪些位置,并确定将对象存放在堆上还是栈
  • 如果对象没有逃逸到方法的外部,可以将其存放在栈上,避免频繁的堆内存分配和垃圾回收,从而提高程序的性能

线上服务 CPU 占用过高怎么排查?

  • 首先找出哪个进程占用 CPU 过高
  • 然后找到进程中的哪个线程占用 CPU 过高
  • 找到线程 ID 后,打印出对应线程的堆栈信息
  • 根据线程的堆栈信息定位到具体代码

内存飙高问题怎么排查?

如果内存飙高发生在 Java 进程上,一般是因为创建了大量的对象,垃圾回收的速度跟不上对象创建的速度,或者是内存泄露导致对象无法回收

举例栈溢出的情况?

  • 栈溢出就是方法执行时,创建的栈帧超过了栈的深度,出现 StackOverflowError
  • 解决方法:使用参数 -Xss 调整 JVM 栈的大小
  • 具体例子:
    • 局部数组过大
    • 递归调用的层次太多。递归函数在运行时会执行压栈操作
    • 指针或数组越界。例如字符串拷贝,处理用户输入

调整栈大小,就能保证不出现溢出吗?

  • 不能
  • 如果程序是死递归的情况,调整栈的大小只是说异常出现的时间会晚一些

分配的栈内存越大越好吗?

  • 不是
  • 如果程序是死递归的情况,分配大内存的栈只是说异常出现的时间会晚一些
  • 会导致可执行的线程数减少,影响其他内存结构

垃圾回收是否会涉及到虚拟机栈

  • 不会,因为只有入栈出栈操作,出栈的过程就相当于 GC

方法中定义的局部变量是否线程安全?

  • 如果只有一个线程才可以操作此数据,则必然线程安全
  • 如果有多个线程操作此数据,则此数据是共享数据。如果不考虑同步机制,会存在线程安全问题
  • 如果变量是在方法内部产生,内部消亡的就是安全的,不是内存产生,或者作为返回值返回的(生命周期没有结束)就不是安全的

静态变量和局部变量?

  • 参数表分配完毕之后,再根据方法体内的变量顺序和作用域分配
  • 类变量表有两次初始化的机会
    • 第一次是在准备阶段,执行系统初始化,对类变量设置零值
    • 第二次是在初始化阶段,赋予程序员在代码中定义的初始值
  • 局部变量不存在系统初始化的过程,即定义了局部变量必须人为的初始化

如何判断一个常量是废弃常量?

  • 假设在字符串常量池中存在字符串 “abc”
  • 如果当前没有任何 String 对象引用该字符串常量的话,就说明常量 “abc” 就是废弃常量

秋招后端开发面试题系列目录
一、Java
1.1 Java基础上
1.2 Java基础下
1.3 Java集合
1.4 JavaIO
1.5 Java多线程上
1.6Java多线程下
二、JVM
2.1 JVM底层原理
2.2 垃圾回收器
2.3 垃圾回收算法
2.4 类加载机制
2.5 运行时数据区
三、MySQL
3.1 MySQL基础
3.2 事务
3.3 索引
3.4 锁机制
3.5 MVCC
四、Redis
4.1 Redis基础
4.2 缓存原理
五、中间件
5.1 RabbitMQ
六、Spring开源框架
6.1 Spring
6.2 Spring MVC
6.3 Spring Boot
6.4 MyBatis
七、操作系统
八、计算机网络
九、设计模式
十、微服务架构
十一、Spring Cloud分布式
11.1 分布式基础
11.2 Spring Cloud
11.3 GateWay
11.4 Nacos
11.5 OpenFeign
11.6 Ribbon
十二、算法
十三、项目

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/830083.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

k8s pod使用sriov

之前的文章中讲了k8s multus的使用&#xff0c;本章节来讲述下如何使用multus来实现sriov的使用。 一、sriov 简介 SR-IOV在2010年左右由Intel提出&#xff0c;但是随着容器技术的推广&#xff0c;intel官方也给出了SR-IOV技术在容器中使用的开源组件&#xff0c;例如&#…

3MF体积设计扩展

3MF 联盟最近宣布了他们最新的体积设计扩展&#xff08;volumetric design extension&#xff09;&#xff0c;用于通过基于体积的描述来编码几何形状和空间多样性属性。 该组织致力于推进 3D 打印的通用规范&#xff0c;目前正在新扩展达到 1.0 之前征求公众反馈。 NSDT工具推…

OpenCV 实现重新映射

返回:OpenCV系列文章目录&#xff08;持续更新中......&#xff09; 上一篇&#xff1a;OpenCV 实现霍夫圆变换 下一篇 :OpenCV实现仿射变换 目标 在本教程中&#xff0c;您将学习如何&#xff1a; 一个。使用 OpenCV 函数 cv&#xff1a;&#xff1a;remap 实现简单的重新…

20240428如何利用IDM下载磁链视频

缘起&#xff1a; https://weibo.com/tv/show/1034:4864336909500449 中国获奖独立纪录片《阿辉》揭秘红灯区“教父”的生存法则 5,751次观看 1年前 发布于 陕西 身为里中横 67.7万粉丝 互联网科技博主 微博原创视频博主 头条文章作者 https://weibo.com/tv/show/1034:4864…

数据通信-A

数据通信 一、数据通信网络基础二、VRP系统三、eNSP配置命令 不是从零开始&#xff0c;有一些基础&#xff0c;主要记录配置命令。一、数据通信网络基础 图标&#xff1a;主要是认识第一行。 常见术语&#xff1a;数据通信网络最基本的功能是实现数据互通。 数据载荷&#…

解决IDEA下springboot项目打包没有主清单属性

1.问题出现在SpringBoot学习中 , 运行maven打包后无法运行 报错为spring_boot01_Demo-0.0.1-SNAPSHOT.jar中没有主清单属性 SpringBoot版本为 2.6.13 Java 版本用的8 解决方法 1.执行clean 删除之前的打包 2.进行打包规范设置 2.1 3.进行问题解决 (借鉴了阿里开发社区) 使用…

[嵌入式系统-53]:嵌入式系统集成开发环境大全

目录 一、嵌入式系统集成开发环境分类 二、由MCU芯片厂家提供的集成开发工具 三、由嵌入式操作提供的集成开发工具 四、由第三方工具厂家提供的集成开发工具 一、嵌入式系统集成开发环境分类 嵌入式系统集成开发工具和集成开发环境可以按照不同的分类方式进行划分&#xff…

【LAMMPS学习】八、基础知识(5.2)粒度模型

8. 基础知识 此部分描述了如何使用 LAMMPS 为用户和开发人员执行各种任务。术语表页面还列出了 MD 术语&#xff0c;以及相应 LAMMPS 手册页的链接。 LAMMPS 源代码分发的 examples 目录中包含的示例输入脚本以及示例脚本页面上突出显示的示例输入脚本还展示了如何设置和运行各…

将针孔模型相机 应用到3DGS

Motivation 3DGS 的 投影采用的是 CG系的投影矩阵 P P P, 默认相机的 principal point (相机光心) 位于图像的中点处。但是 实际应用的 绝大多数的 相机 并不满足这样一个设定&#xff0c; 因此我们 需要根据 f , c x , c y {f,c_x, c_y} f,cx​,cy​ 这几个参数重新构建3D …

centos 7 yum install -y nagios

centos 7 systemctl disable firewalld --now vi /etc/selinux/config SELINUXdisabled yum install -y epel-release httpd nagios yum install -y httpd nagios systemctl enable httpd --now systemctl enable nagios --now 浏览器 IP/nagios 用户名&#xff1a;…

vue学习的预备知识为学好vue打好基础

目录 Vue是什么 &#xff1f;如何使用Vue &#xff1f;Vue ApiVue入口apiVue实例apiVue函数api 无构建过程的渐进式增强静态HTMLVue模块化构建工具npmyarnWebpackvue-cliVite Vue是什么 &#xff1f; 文章基于Vue3叙述。 Vue (发音为 /vjuː/&#xff0c;类似 view) 是一款用于…

十大USDT交易平台大全XEX交易所

USDT是一种基于比特币区块链网络的加密代币&#xff0c;主要运用于数字货币交易平台&#xff0c;以稳定币为主。USDT的核心价值在于其与真实货币的固定兑换比率1:1&#xff0c;所以被称为Tether。随着加密货币市场的不断壮大&#xff0c;越来越多的交易平台开始支持USDT&#x…

2024深圳杯(东北三省)数学建模C题完整论文讲解(含完整python代码及所有残骸音爆位置求解结果)

大家好呀&#xff0c;从发布赛题一直到现在&#xff0c;总算完成了2024深圳杯&#xff08;东北三省数学建模联赛&#xff09;A题多个火箭残骸的准确定位完整的成品论文。 本论文可以保证原创&#xff0c;保证高质量。绝不是随便引用一大堆模型和代码复制粘贴进来完全没有应用糊…

【vscode环境配置系列】vscode远程debug配置

VSCODE debug环境配置 插件安装配置文件debug 插件安装 安装C/C, C/C Runner 配置文件 在项目下建立.vscode文件夹&#xff0c;然后分别建立c_cpp_properties.json&#xff0c; launch.json&#xff0c;tasks.json&#xff0c;内容如下&#xff1a; c_cpp_properties.json:…

如何解决pycharm创建项目报错 Error occurred when installing package ‘requests‘. Details.

&#x1f42f; 如何解决PyCharm创建项目时的包安装错误&#xff1a;‘requests’ &#x1f6e0;️ 文章目录 &#x1f42f; 如何解决PyCharm创建项目时的包安装错误&#xff1a;requests &#x1f6e0;️摘要引言正文&#x1f4d8; **问题分析**&#x1f680; **更换Python版本…

如何利用快解析软件搭建映射端口

端口映射&#xff0c;就是将内网中主机的一个端口映射到外网主机的一个端口&#xff0c;提供相应的服务&#xff0c;当用户访问外网IP的这个端口时&#xff0c;服务器自动将请求映射到对应局域网内部的机器上。如何才能实现端口映射&#xff1f;今天小编给大家介绍两种方法&…

fetch请求后端返回文件流,并下载。

前端&#xff1a; <script src"~/layui/layui.js"></script> <script src"~/Content/js/common/js/vue.min.js"></script> <script src"~/Content/js/common/js/jquery-1.10.2.min.js"></script><styl…

QT学习篇—qt软件安装

qt下载网址http://download.qt.io/new_archive/qt/ QT官网Qt | Tools for Each Stage of Software Development LifecycleAll the essential Qt tools for all stages of Software Development Lifecycle: planning, design, development, testing, and deployment.https:…

虚拟机扩容-根目录挂载sda1的空间不足

提醒&#xff01;不管成不成功&#xff0c;一定要先备份一份虚拟机&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 走过路过点个关注吧&#xff0c;想到500粉丝&#xff0c;哭。一、查看分区情况 df -h可以看到/dev/sda1已经被占满了 2.关闭虚拟机&#xff…

TinyShell后门通信模型剖析

TinyShell后门通信模型剖析 通过对TinyShell后门的外联通信函数进行剖析&#xff0c;梳理其通信过程如下&#xff1a; 调用gettimeofday函数及getpid函数获取当前时间tv及进程pid&#xff0c;将tv和pid作为SHA1算法的输入&#xff0c;生成得到20字节的IV1数据调用gettimeofda…