10个经典又容易被人疏忽的JVM面试题,java面试刷题网站

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024b (备注Java)
img

正文

//StringBuilder可能被其他方法改变,逃逸到了方法外部。

public StringBuilder  escape(String a, String b) {

//公众号:捡田螺的小男孩

StringBuilder str = new StringBuilder();

str.append(a);

str.append(b);

return str;

}

//不直接返回StringBuffer,不发生逃逸

public String notEscape(String a, String b) {

//公众号:捡田螺的小男孩

StringBuilder str = new StringBuilder();

str.append(a);

str.append(b);

return str.toString();

}

//外部线程可见object,发生逃逸

public void objectEscape(){

object = new Object();

}

//仅方法内部可见,不发生逃逸

public void objectNotEscape(){

Object object = new Object();

}

}

「逃逸分析的好处」

栈上分配,可以降低垃圾收集器运行的频率。同步消除,如果发现某个对象只能从一个线程可访问,那么在这个对象上的操作可以不需要同步。标量替换,把对象分解成一个个基本类型,并且内存分配不再是分配在堆上,而是分配在栈上。这样的好处有,一、减少内存使用,因为不用生成对象头。二、程序内存回收效率高,并且GC频率也会减少。❞

2.虚拟机为什么使用元空间替换了永久代?

====================

「什么是元空间?什么是永久代?为什么用元空间代替永久代?」 我们先回顾一下**「方法区」**吧,看看虚拟机运行时数据内存图,如下:

10个经典又容易被人疏忽的JVM面试题

方法区和堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据。

「什么是永久代?它和方法区有什么关系呢?」

如果在HotSpot虚拟机上开发、部署,很多程序员都把方法区称作永久代。可以说方法区是规范,永久代是Hotspot针对该规范进行的实现。在Java7及以前的版本,方法区都是永久代实现的。

「什么是元空间?它和方法区有什么关系呢?」

对于Java8,HotSpots取消了永久代,取而代之的是元空间(Metaspace)。换句话说,就是方法区还是在的,只是实现变了,从永久代变为元空间了。

「为什么使用元空间替换了永久代?」

  • 永久代的方法区,和堆使用的物理内存是连续的。

10个经典又容易被人疏忽的JVM面试题

**「永久代」**是通过以下这两个参数配置大小的~

  • -XX:PremSize:设置永久代的初始大小

  • -XX:MaxPermSize: 设置永久代的最大值,默认是64M

对于**「永久代」,如果动态生成很多class的话,就很可能出现「java.lang.OutOfMemoryError: PermGen space错误」**,因为永久代空间配置有限嘛。最典型的场景是,在web开发比较多jsp页面的时候。

  • JDK8之后,方法区存在于元空间(Metaspace)。物理内存不再与堆连续,而是直接存在于本地内存中,理论上机器**「内存有多大,元空间就有多大」**。

10个经典又容易被人疏忽的JVM面试题

可以通过以下的参数来设置元空间的大小:

-XX:MetaspaceSize,初始空间大小,达到该值就会触发垃圾收集进行类型卸载,同时GC会对该值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,那么在不超过MaxMetaspaceSize时,适当提高该值。-XX:MaxMetaspaceSize,最大空间,默认是没有限制的。-XX:MinMetaspaceFreeRatio,在GC之后,最小的Metaspace剩余空间容量的百分比,减少为分配空间所导致的垃圾收集-XX:MaxMetaspaceFreeRatio,在GC之后,最大的Metaspace剩余空间容量的百分比,减少为释放空间所导致的垃圾收集❞

「所以,为什么使用元空间替换永久代?」

表面上看是为了避免OOM异常。因为通常使用PermSize和MaxPermSize设置永久代的大小就决定了永久代的上限,但是不是总能知道应该设置为多大合适, 如果使用默认值很容易遇到OOM错误。当使用元空间时,可以加载多少类的元数据就不再由MaxPermSize控制, 而由系统的实际可用空间来控制啦。

3.什么是Stop The World ? 什么是OopMap?什么是安全点?

=======================================

进行垃圾回收的过程中,会涉及对象的移动。为了保证对象引用更新的正确性,必须暂停所有的用户线程,像这样的停顿,虚拟机设计者形象描述为**「Stop The World」**。

在HotSpot中,有个数据结构(映射表)称为**「OopMap」。一旦类加载动作完成的时候,HotSpot就会把对象内什么偏移量上是什么类型的数据计算出来,记录到OopMap。在即时编译过程中,也会在「特定的位置」**生成 OopMap,记录下栈上和寄存器里哪些位置是引用。

这些特定的位置主要在:

  • 1.循环的末尾(非 counted 循环)

  • 2.方法临返回前 / 调用方法的call指令后

  • 3.可能抛异常的位置

这些位置就叫作**「安全点(safepoint)。」** 用户程序执行时并非在代码指令流的任意位置都能够在停顿下来开始垃圾收集,而是必须是执行到安全点才能够暂停。

4.说一下JVM 的主要组成部分及其作用?

=====================

10个经典又容易被人疏忽的JVM面试题

JVM包含两个子系统和两个组件,分别为

Class loader(类装载子系统)Execution engine(执行引擎子系统);Runtime data area(运行时数据区组件)Native Interface(本地接口组件)。❞

  • 「Class loader(类装载):」 根据给定的全限定名类名(如:java.lang.Object)来装载class文件到运行时数据区的方法区中。

  • 「Execution engine(执行引擎)」:执行class的指令。

  • 「Native Interface(本地接口):」 与native lib交互,是其它编程语言交互的接口。

  • 「Runtime data area(运行时数据区域)」:即我们常说的JVM的内存。

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

5. 守护线程是什么?守护线程和非守护线程的区别是?守护线程的作用是?

====================================

「守护线程」是区别于用户线程哈,「用户线程」即我们手动创建的线程,而守护线程是程序运行的时候在后台提供一种「通用服务的线程」。垃圾回收线程就是典型的守护线程。

「守护线程和非守护线程的区别是?」 我们通过例子来看吧~

/**

* 关注公众号:捡田螺的小男孩

*/

public static void main(String[] args) throws InterruptedException {

Thread t1 = new Thread(()-> {

while (true) {

try {

Thread.sleep(1000);

System.out.println(“我是子线程(用户线程.I am running”);

} catch (Exception e) {

}

}

});

//标记为守护线程

t1.setDaemon(true);

//启动线程

t1.start();

Thread.sleep(3000);

System.out.println(“主线程执行完毕…”);

}

运行结果:

10个经典又容易被人疏忽的JVM面试题

可以发现标记为守护线程后,「主线程销毁停止,守护线程一起销毁」。我们再看下,去掉 t1.setDaemon(true)守护标记的效果:

public static void main(String[] args) throws InterruptedException {

Thread t1 = new Thread(()-> {

while (true) {

try {

Thread.sleep(1000);

System.out.println(“我是子线程(用户线程.I am running”);

} catch (Exception e) {

}

}

});

//启动线程

t1.start();

Thread.sleep(3000);

System.out.println(“主线程执行完毕…”);

}

10个经典又容易被人疏忽的JVM面试题

所以,当主线程退出时,JVM 也跟着退出运行,守护线程同时也会被回收,即使是死循环。如果是用户线程,它会一直停在死循环跑。这就是**「守护线程和非守护线程的区别」**啦。

守护线程拥有**「自动结束自己生命周期的特性」,非守护线程却没有。如果垃圾回收线程是非守护线程,当JVM 要退出时,由于垃圾回收线程还在运行着,导致程序无法退出,这就很尴尬。这就是「为什么垃圾回收线程需要是守护线程啦」**。

6.WeakHashMap了解过嘛?它是怎么工作的?

==========================

「WeakHashMap」 类似HashMap ,不同点在WeakHashMap的key是**「弱引用」**的key。

谈到**「弱引用」**,在这里回顾下四种引用吧

强引用:Object obj=new Object()这种,只要强引用关系还存在,垃圾收集器就永远不会回收掉被引用的对象。软引用: 一般情况不会回收,如果内存不够要溢出时才会进行回收弱引用:当垃圾收集器开始工作,无论当前内存是否足够,都会回收掉只被弱引用关联的对象。虚引用:为一个对象设置虚引用的唯一目的只是为了能在这个对象被回收时收到一个系统的通知。❞

正是因为WeakHashMap使用的是弱引用,「它的对象可能随时被回收」。WeakHashMap 类的行为部分**「取决于垃圾回收器的动作」,调用两次size()方法返回不同值,调用两次isEmpty(),一次返回true,一次返回false都是「可能的」**。

WeakHashMap**「工作原理」**回答这两点:

WeakHashMap具有弱引用的特点:随时被回收对象。

发生GC时,WeakHashMap是如何将Entry移除的呢?❞

WeakHashMap内部的Entry继承了WeakReference,即弱引用,所以就具有了弱引用的特点,「随时可能被回收」。看下源码哈:

private static class Entry<K,V> extends WeakReference implements Map.Entry<K,V> {

V value;

final int hash;

Entry<K,V> next;

/**

* Creates new entry.

*/

Entry(Object key, V value,

ReferenceQueue queue,

int hash, Entry<K,V> next) {

super(key, queue);

this.value = value;

this.hash  = hash;

this.next  = next;

最后

手绘了下图所示的kafka知识大纲流程图(xmind文件不能上传,导出图片展现),但都可提供源文件给每位爱学习的朋友

image.png

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
ntry<K,V> next) {

super(key, queue);

this.value = value;

this.hash  = hash;

this.next  = next;

最后

手绘了下图所示的kafka知识大纲流程图(xmind文件不能上传,导出图片展现),但都可提供源文件给每位爱学习的朋友

[外链图片转存中…(img-lvWGoGK5-1713375723114)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
[外链图片转存中…(img-nlESDlDj-1713375723114)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 15
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值