当前位置: 首页 > news >正文

JVM高频面试题

1、项目中什么情况下会内存溢出,怎么解决?

	(1)误用固定大小线程池导致内存溢出  Excutors.newFixedThreadPool内最大线程数是21亿(2) 误用带缓冲线程池导致内存溢出最大线程数是21亿(3)一次查询太多的数据,导致内存占用太大(4)动态生成类导致内存溢出

2、类加载过程?

三个过程:加载:

在这里插入图片描述
链接:
在这里插入图片描述
链接又分为:验证、准备、解析

	初始化:

在这里插入图片描述
总结:所谓类加载机制就是:虚拟机把Class文件加载到内存,并对数据进行校验、转换解析和初始化,形成虚拟机可以直接使用的Java类型,即java.lang.class

补充:类.class只会触发类加载且只会加载一次,而new 类()则会触发类初始化。静态变量的赋值在初始化时完成。

3、什么是双亲委派?
在这里插入图片描述

4.对象引用分为哪几类?

	1.强引用

在这里插入图片描述
正常对象引用

		2.软引用

在这里插入图片描述
系统内存足够时不会被回收,不足时才会被回收。且若回收之后仍内存不够,则会内存溢出
3.弱引用
在这里插入图片描述
无论内存是否充足都会被回收
4.虚引用
在这里插入图片描述
主要用来跟踪对象被垃圾回收的过程,任何时候都可能被垃圾回收

5、堆(heap)和栈(stack)有什么区别?
1、内存分配方式不同:堆是动态分配内存空间的;栈是静态分配内存空间的。
2、内存空间大小不同:堆的内存空间通常比较大,可以动态扩展;而栈的内存空间通常比较小,由于静态分配,所以大小固定。
3、内存分配效率不同:堆的内存分配效率相对较低,因为需要进行垃圾回收和内存整理等操作;而栈的内存分配效率相对较高,因为只需要简单地移动指针即可。
4、存储内容不同:堆用于存储对象实例和数组等动态数据,它具有很好的灵活性;而栈用于存储方法调用的局部变量和操作数栈等数据,它具有很好的局部性和快速访问的特点。

6、什么情况下会发生栈内存溢出?

方法调用层次过深:如果方法调用层次过深,每次方法调用都会在栈中创建一个新的栈帧,如果栈空间不足,就会导致栈内存溢出。
局部变量过多:如果方法中定义了大量的局部变量,也会占用栈空间,如果栈空间不足,就会导致栈内存溢出。
递归调用:递归调用会不断地向栈中添加新的方法调用,如果递归深度过大,就会导致栈内存溢出。
参数传递过多:如果方法参数传递过多,也会占用栈空间,如果栈空间不足,就会导致栈内存溢出。
大量线程调用:如果同时有大量的线程在栈中进行方法调用,也会占用栈空间,如果栈空间不足,就会导致栈内存溢出。

7、什么是OOM?
OOM 是 OutOfMemoryError 的缩写,是指在 Java 程序运行期间,由于内存不足而导致程序出现错误的情况。OOM 错误通常分为堆内存溢出和非堆内存溢出两种情况。

8、如何判断一个对象是否存活?

	1、引用计数法:引用计数法是一种简单的垃圾回收算法,它通过统计对象的引用计数,来判断对象是否存活。每当一个对象被引用时,它的引用计数加 1,当一个对象的引用计数为 0 时,说明该对象已经不再被使用,可以被垃圾回收器回收。但是,引用计数法无法处理循环引用的情况,即两个或多个对象相互引用,导致它们的引用计数都不为 0,无法被回收。2、可达性分析法:可达性分析法是一种常用的垃圾回收算法,它通过从一组根对象开始,查找所有被这组根对象所引用的对象,以此来确定哪些对象是存活的。在可达性分析法中,根对象可以是程序中的静态变量、本地变量或者正在执行的线程等。当一个对象无法被根对象所引用时,说明该对象已经不再被使用,可以被垃圾回收器回收。

可达性分析法是 Java 中常用的判断对象存活的方式,它考虑了对象之间的引用关系,可以处理循环引用的情况,而引用计数法则无法处理循环引用的情况,因此在 Java 中并不采用引用计数法来判断对象是否存活。

9、有哪几种垃圾回收器?各自的优点是什么?
Serial 垃圾回收器:是一种单线程的垃圾回收器,它使用标记-清除算法来回收垃圾对象。优点是简单高效,适合小型应用场景,但是在多核处理器上的表现较差。

Parallel 垃圾回收器:是一种多线程的垃圾回收器,它使用标记-整理算法来回收垃圾对象。优点是在多核处理器上的表现比 Serial 垃圾回收器更好,适合中型应用场景。

CMS 垃圾回收器:是一种并发的垃圾回收器,它使用标记-清除算法来回收垃圾对象。优点是回收效率高,暂停时间短,适合大型应用场景。但是,CMS 垃圾回收器会产生内存碎片,可能会导致频繁的 Full GC,影响应用性能。

G1 垃圾回收器:是一种并发的垃圾回收器,它使用标记-整理算法来回收垃圾对象。优点是在大型应用场景下表现良好,可以有效避免内存碎片和频繁的 Full GC。同时,G1 垃圾回收器还支持设置可预测的暂停时间,使得应用程序的性能得到进一步优化。可以优先回收大块垃圾的区域

Serial和CMS都是标记清楚,Parallel和G1都是标记整理。G1现在用的最多。

10、什么是JVM内存结构?
JVM分为堆、虚拟机栈、本地方法栈、程序计数器、方法区

虚拟机栈里有局部变量表、操作数栈、动态链接、方法返回信息

本地方法栈:是C栈,执行本地方法

程序计数器:用于记录当前虚拟机正执行的线程指令地址

堆:所有线程共享的一块内存,虚拟机开启时就创建,可通过GC进行回收。

方法区:jdk1.8之后叫元数据区

http://www.lryc.cn/news/40674.html

相关文章:

  • Windows环境下实现设计模式——状态模式(JAVA版)
  • 【总结】多个条件排序(pii/struct/bool)
  • 基于stm32mp157 linux开发板ARM裸机开发教程Cortex-A7 开发环境搭建(连载中)
  • 最适合游戏开发的语言是什么?
  • C语言刷题(7)(字符串旋转问题)——“C”
  • 有趣且重要的JS知识合集(18)浏览器实现前端录音功能
  • 面试官:聊聊你知道的跨域解决方案
  • SpringCloud五大核心组件
  • Verilog HDL语言入门(二)
  • Simpleperf详细使用
  • 【算法基础】二分图(染色法 匈牙利算法)
  • Caputo 分数阶微分方程-慢扩散方程初边值问题基于L1 逼近的空间二阶方法及其Matlab程序实现
  • I.MX6ULL_Linux_驱动篇(29) GPIO驱动
  • jupyter的安装和使用
  • Springboot新手开发 Cloud篇
  • Linux:函数指针做函数参数
  • Vue3(递归组件) + 原生Table 实现树结构复杂表格
  • ArrayList底层源码解析
  • python:DIY字符画的程序使用说明.doc
  • 【Python/Opencv】图像权重加法函数:cv2.addWeighted()详解
  • 容器的老祖宗LXC和Docker的关系
  • Webpack迁移Rspack速攻实战教程(前瞻版)
  • 一行代码“黑”掉任意网站
  • 51单片机入门 -驱动 8x8 LED 点阵屏
  • Xinlinx zynq7045国产替代 FMQL45T900全国产化 ARM 核心板+扩展板
  • 硬刚ChatGPT!文心一言能否为百度止颓?中国版ChatGPT“狂飙”的机会在哪儿?
  • Python 异步: 在非阻塞子进程中运行命令(19)
  • 蓝桥杯嵌入式第五课--输入捕获
  • Spring事务和事务传播机制
  • 基于OpenCV+CUDA实时视频抠绿、背景合成以及抠绿算法小结