JAVA面试-调优篇

一、说一说JAVA内存模型?

在这里插入图片描述

二、JAVA类的加载过程是怎样的?什么是双亲委派机制?有什么作用?一个对象从加载到JVM,再到被GC清楚,都经历过什么过程?

JAVA的类加载过程:

AppClassLoader ----> ExtClassLoader --->BootStrapClassLoader

每种类加载器都有自己的加载目录

JAVA中的类加载器继承关系:

ExtClassLoader ----> URLClassLoader ---> SecureClassLoader ----> ClassLoader

每个类加载器对它加载的过的类,都是有一个缓存的

双亲委派:

向上委托查找,向下委托加载
在这里插入图片描述
双亲委派的作用:保证 JAVA 底层的类不会被应用程序覆盖(自己写的同名类)

类加载过程:加载 —> 连接 —> 初始化

加载:把 JAVA 字节码数据加载到 JVM 内存中,并映射成JVM认可的数据结构
连接:分为三个小阶段:
1、验证:检查加载的字节信息是否符合JVM规范
2、准备:创建类或接口的静态变量,并赋予初始值,半初始化状态
3、解析:把符号引用转为直接引用
初始化:执行自己写的代码

对接生命周期:

1、用户创建一个对象,JVM 首先需要到方法区去找对象的类型信息,然后再创建对象
2、JVM 要实例化一个对象,首先要在堆中创建一个对象,即半初始化状态
3、对象首先会分配在堆内存的新生代的 Eden 区,然后经过一次 MinorGC, 对象如果存活,就会进入 S 区, 在后续的每次 GC 中,如果对象一直存活,就会在S qu区来回拷贝,没移动一次,频率加1,直到移入老年代。(默认是15次)
4、当方法执行结束后,栈中的指针会先移除掉
5、堆中的对象,经过 FullGC,就会标记为垃圾,然后被GC线程清理掉

三、怎么样缺点一个对象是不是垃圾?什么是GCRoot?

有两种定位垃圾的方式:

1、引用计数:这种方式是给堆内存中的每个对象记录一个引用个数,引用个数为0的就认为是垃圾。这种方式无法解决循环引用的问题,有内存泄漏的问题存在。
2、根可达算法:这种方式在内存中,从引用根对象向下一直找对象,找不到的对象就是垃圾

有哪些GCRoot ? stack --> JVM stack Native stack --> class 类 run-time coustant pool 常量池 static reference 静态变量

四、JVM有哪些垃圾回收算法?

MarkSweep 标记清除算法

标记阶段:把垃圾内存标记出来
清理阶段:直接将垃圾内存回收;

算法简单,但是有一个严重的问题,就是会产生大量的内存碎片

Copying 拷贝算法

为了解决标记清除算法的内存碎片问题,就产生了拷贝算法。将内存分为大小相同的两半,每次只使用其中的一半,垃圾回收时,将当前这一块的存活对象全部拷贝到另一半,然后当前这一半就可以直接清楚了。
这种算法没有内存碎片,但是它的问题就是在于浪费空间,而且,它的效率也跟存活对象的数量有关

MarkCompack 标记压缩算法

为了解决拷贝算法的缺陷,就提出了标记压缩算法。这种算法在标记阶段和标清除是一样的,但是在完成标记之后,不说直接清理垃圾内存,而是将存活的对象往另一端移动,然后将端边界以外的所有内存直接清理掉

这三种算法各有利弊,各自有各自的使用场景

五、JVM有哪些垃圾回收器?它们都是怎样工作的?什么是STW?它们都发生在哪些阶段?什么是三色标记?如何解决错标记和漏标记的问题?为社么要设计这么多垃圾回收器?

STW : stop-the-world 是在垃圾回收算法执行过程中,需要将 JVM 内存冻结的一种状态。在 STW 状态下,JAVA 的所有线程都是停止状态(GC线程除外),native 方法可以执行,但是它们不能与 JVM 交互。GC 各种算法优化的重点,就是减少 STW,同时这也是 JVM 调优的重点

JVM 垃圾回收器:
分代算法:1、年轻代
2、老年代
不分代算法

Serial 串行:整个过程比较简单,需要GC时,直接暂停,GC完了再继续。是早期的来及回收算法,只有一个线程执行GC,在多CPU架构下,性能就会严重下降,只适用于几十兆的内存空间

Paraller 并行:在串行的基础上,增加多线程GC,PS + PO 这种组合是JDK1.8默认的垃圾回收器,在多CPU架构下,性能比 Serial 高很多

CMS concurrent Mark Sweep 并行标记清除:核心思想就是将STW打散,让一部分GC线程与用户线程并发执行,整改GC过程分为四个阶段:
1、初始化标记阶段:STW只标记出根对象直接引用的对象
2、并发标记阶段:继续标记其他对象,与应用程序并发执行
3、重新标记阶段:STW对并发执行阶段的对象进行重新标记
4、并发清理阶段:并行将产生的垃圾清除,清除过程中,应用程序又会产生新的垃圾,叫浮动垃圾,这些垃圾到下一次GC过程中清理

G1 Garbage Fist 垃圾优先:它的内存模型是实际不分代,但是逻辑上是分代的。在内存模型中,对于堆内存就不再分新生代和老年代了,而是划分为一个一个小内存块,叫做 Region。每个 Region 可以隶属于不同的年代

G1 GC 分为五个阶段
1、初始标记:标记出 GCRoot 直接引用的对象
2、标记 Region:通过 RSet标记出上一阶段的 Region 引用到的 Old 区 Region
3、并发标记:跟 CMS 的步骤差不多,只是遍历的范围不再是这弓 Old 区,而只需遍历第二步标记出来的 Rgion
4、重新标记:根 CMS 中重新标记的过程差不多
5、垃圾清理:跟 CMS 不同的是,G1 采用拷贝算法,直接将整改 Region 中的对象拷贝到另一个 Region 中,而这个阶段,G1 只选择较多的 Region 清理,并不是完全清理

之所有设计这么多垃圾回收器,是因为内存在逐渐变大

三色标记:CMS 的核心算法就是三色标记,是一种逻辑上的抽象,将每个内存对象分成三种颜色
黑色:表示自己和成用变量都已标记完毕
灰色:自己标记完了,但是成用变量还没完全标记完
白色:自己未标记完

CMS 通过增加标记 increment update 的方式来解决漏标记的问题

六、如何进行JVM调优?JVM参数有哪些?怎么查看一个JAVA进程的JVM参数?

JVM 调优主要就是通过定制 JVM 运行参数来提高 JAVA 应用程序的运行速度
JVM 参数大致分为三类:
1、标准指令: - 开头,这些是所有的 Hotspot 都支持的参数,可以用 java -help 打印出来
2、非标准指令: -X 开头,这些指令通常是跟特定的 Hotspot 版本对应的, java -X打印出来
3、不稳定指令:-XX 开头,这一类参数跟特定的 Hotspot 版本对应的,并且变化非常大,详细文档资料非常少

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值