JVM面试题

JVM 的运行时数据区是如何划分的?
  • 程序计数器:记录当前线程执行的字节码行号,是线程私有的,主要用于线程切换后能恢复正确的执行位置。
  • 虚拟机栈:每个线程都有自己的虚拟机栈,栈中存储着栈帧,每个栈帧对应一个方法的调用,包含局部变量表、操作数栈、动态链接、方法出口等信息。
  • 本地方法栈:与虚拟机栈类似,不过它是为本地方法服务的。
  • 堆:是 JVM 中最大的一块内存区域,被所有线程共享,用于存储对象实例和数组。
  • 方法区:也是所有线程共享的区域,用于存储类的元数据(如类的结构、字段、方法、常量池等)。在 Java 8 及以后,方法区被元空间(Metaspace)取代。
什么是垃圾回收? 垃圾回收有哪些算法?

它会自动检测并回收不再使用的对象所占用的内存空间

标记 - 清除算法: 先标记出需要回收的对象, 然后统一回收, 会产生大量内存碎片

标记 - 整理算法: 先标记出需要回收的对象, 然后将存活的对象向一端移动, 最后清除掉端边界以外的内存, 避免了内存碎片问题

复制算法: 将内存分成相等的两块,每次只使用其中的一块,当这一块内存用完后,将存活的对象复制到另一块上,然后清理当前使用的这块内存. 这种算法实现简单, 效率高, 但是会浪费一半的内存空间.

分代收集算法: 根据对象的存活周期将内存划分为不同的区域(新生代和老年代),针对不同区域的特点采用不同的垃圾回收算法. 例如,新生代存活时间短, 采用复制算法, 老年代对象存活时间长, 采用标记 - 清除 或标记 - 整理算法.

简述 Java 的类加载机制

Java的类加载机制采用双亲委派模型, 当一个类加载器收到类加载请求时, 他首先不会自己去尝试加载这个类, 而是把请求委派给父亲加载器去完成, 每一个层次的类加载器都是如此, 因此所有的加载请求都会传送到顶层的启动类加载器中, 只有当父类加载器反馈自己无法完成该加载请求(它的搜索范围中没有找到所需的类) 时, 子加载器才会尝试自己去加载.

双亲委派模型的特点

1.避免类的重复加载: 如果多个类加载器都去加载同一个类, 会导致内存中存在多个相同的类的实例,会造成资源浪费. 通过双亲委派模型, 同一个类只会被加载一次.

2.保证Java核心类库的安全性: 例如, 用户自定义的java.lang.String类不会被加载,因为启动类加载器会优先加载JDK中的java.lang.String类, 从而避免了恶意代码通过自定义核心类来破坏系统的安全性.

类加载的过程

类加载器的层次结构
### 关于 JVM 面试题的常见问题及其解析 #### 1. **JVM 内存结构** JVM 的内存主要分为堆、方法区、栈、本地方法栈以及程序计数器五个部分。其中,堆是垃圾回收的主要区域,被划分为新生代和老年代[^1]。 ```java // 新生代中的对象分配演示 public class ObjectAllocation { public static void main(String[] args) { byte[] allocation1, allocation2; allocation1 = new byte[2 * 1024 * 1024]; // 对象可能进入 Eden 区 allocation2 = new byte[2 * 1024 * 1024]; // 如果空间不足,则触发 Minor GC } } ``` #### 2. **GC 过程与算法** 常见的垃圾收集算法有标记-清除法、复制法、标记-整理法以及分代收集法。在实际应用中,通常会采用 CMS 或 G1 收集器来优化性能[^1]。 #### 3. **类加载机制** JVM 类加载过程包括加载、验证、准备、解析和初始化阶段。双亲委派模型确保了 Java 核心库的安全性和一致性。 #### 4. **引用类型** Java 中存在四种不同的引用类型:强引用、软引用、弱引用和虚引用。这些引用类型主要用于管理内存资源并配合垃圾回收器工作[^5]。 #### 5. **Happens-Before 原则** Java 提供了一套 Happens-Before 规则以保障线程间的可见性。其中包括程序顺序规则、监视器锁规则、volatile 变量规则等八大原则[^3]。 #### 6. **循环引用问题** 即使两个对象相互持有对方的引用,在它们不再被任何外部变量引用的情况下,现代垃圾回收器仍然能够识别这种关系并将两者都回收掉[^2]。 ```java // 循环引用示例 class Node { private Node next; public Node(Node n) { this.next = n; } protected void finalize() throws Throwable { System.out.println("Finalizing..."); } } public class CircularReferenceTest { public static void main(String[] args) throws InterruptedException { Node nodeA = new Node(null); Node nodeB = new Node(nodeA); nodeA.next = nodeB; // 创建循环引用 nodeB = null; nodeA = null; System.gc(); Thread.sleep(100); // 等待垃圾回收完成 } } ``` #### 7. **JVM 参数调优** 通过设置 `-Xms` 和 `-Xmx` 来指定初始堆大小和最大堆大小;利用 `-XX:+UseG1GC` 开启 G1 垃圾收集器可以有效提升大型应用程序的表现[^4]。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Hsifles

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值