一、什么情况下回法神栈内存溢出?
- 栈是线程私有的,栈的生命周期和线程一样,每个方法在执行的时候就会创建一个栈帧,它包含局部变量表、操作数栈、动态链接、方法出口等信息,局部变量表又包括基本数据类型和对象的引用;
- 当线程请求的栈深度超过了虚拟机允许的最大深度时,会抛出stackOverFlowError异常,方法递归调用可能会出现该问题;
- 调整参数-xss去调整jvm的栈的大小
二、详解JVM内存模型?
jvm将虚拟机分为5大区域:程序计数器、虚拟机栈、本地方法栈、java堆,方法区
程序计数器
线程私有的,是一块很小的内存空间,作为当前线程的行号指示器,用于记录当前虚拟机正在执行的线程指令地址;
虚拟机栈
线程私有的,每个方法执行的时候都会创建一个栈帧,用于存储局部变量表、操作数、动态链接和方法返回的信息等,当线程请求的栈深度超过了虚拟机允许的最大深度时,会抛出栈内存溢出溢出
本地方法栈
线程私有的,保存的是native方法的信息,当一个jvm创建的线程调用native方法后,jvm不会在虚拟机栈中为该线程创建栈帧,而是简单的动态链接并直接调用该方法。
堆
java堆是所有线程共享的一块内存,几乎所有对象的实例和数组都要在堆上分配内存,因此该区域经常发生垃圾回收的操作
方法区
存放已被加载的类信息、常量、静态变量,即时编译器编译后的代码数据。即永久代,在jdk1.8中已经不存在方法区了,被元数据去替代了,原方法被分成两个部分:1.加载类的信息,2.运行时常量池。加载的类的信息被保存在了元数据区中,运行时常量池保存在堆中。
三、JVM中一次完整的GC是什么样子的?对象如何晋升到老年代?
java堆=新生代+老年代;新生代=Eden+Suivivor(S0+S1),默认分配比例是8:1:1,当Eden区空间满了的时候,就会触发一次Minor GC,以收集新生代的垃圾,存活下来的对象会被分配到Survivor区大对象(需要大量连续内存空间的对象)。如果对象在Eden中出生,并且在经历过一次GC只会仍然存活,被分配到存活区的话,年龄+1,此后每经历过一次Minor GC并且存活下来,年龄就+1,当年龄达到15的时候,会被晋升到老年代;当老年代满了,而无法容纳更多对象的话,就会触发一次full gc,full gc存储的是整个内存堆(包括年轻代和老年代);major GC是发生在老年代的GC,清理老年区,经常会伴随至少一次minor gc;
四、Java中的垃圾回收算法
java 中有四种垃圾回收算法,分别是:
1.标记清除法、标记整理法、复制算法、分代收集算法
标记清楚法
第一步:利用可达性去比那里内存,把存活对象和垃圾对象进行标记;
第二步:在遍历一遍,将所有标记的对象回收掉。特点:效率较差,标记和清除的效率都不高;标记和清除后会产生大量的不连续的空间分片,可能会导致只会程序运行的时候需分配大对象而找不到连续分片而不得不触发一次GC
标记整理法
第一步:利用可达性去遍历内存,把存活对象和垃圾对象进行标记;
第二步:将所有存活的对象向一端移动,将端边界以外的对象都回收掉;特点:适用于于存活对象多,垃圾少的情况;需要整理的过程,无空间碎片产生。
复制算法
将内存按照容量大小分为大小相等的两块,每次只使用一块,当一块使用完了,就将还存活的对象移到另一块上,然后把使用过的内存空间移除;特点:不会产生空间随便;内存使用率极低;
分代收集算法
根据内存对象的存活周期不同,将内存划分成几块,java虚拟机一般将内存分成新生代和老年代,在新生代中,有大量对象死去和少量对象存活,所以采用复制算法,只需要付出少量存活对象的复制成功就可以完成收集;老年代中因为对象的存活率极高,没有额外的空间对他进行分配担保,所以采用标记清理或者标记整理算法进行回收
五、如何判断一个对象是否存活
判断一个对象是否存活,分为两种算法:
- 引用计数法
- 可达性分析算法
引用计数法
给每一个对象设置一个引用计数器,当有一个地方引用该对象的时候,引用计数器就+1,引用失效时,引用计数器就-1,当引用计数器为0时,就说明这个对象没有被引用,也就是垃圾对象,等待回收;缺点:无法解决循环引用的问题,当A引用B,B也引用A的时候,此时对AB对象的引用都不为0,此时也就无法垃圾回收,所以一半主流虚拟机都不采用这个方法;
可达性分析法
从一个被称为GC ROOTS的对象向下搜索,如果一个对象到GC ROOTS没有任何引用链相连接时,说明此对象不可用,在java中可以作为GC ROOTS的对象有以下几种;
虚拟机栈中引用的对象、方法区类静态属性、引用的变量方法区、常量池引用的对象、本地方法栈JNI引用的对象,当一个对象满足上述条件的时候,不会马上被回收,还需要进行两次标记;第一次标记:判断当前对象是否有finalize()方法并且该方法没有被执行过,若不存在则标记为垃圾对象,等待回收;若有的话,则进行第二次标记;第二次标记当前对象放入F-Queue队列,并生成一个finalize线程去执行该方法,虚拟机不保证该方法一定会被执行,这是因为如果线程执行缓慢或者进入了死锁,会导致回收系统的崩溃;如果执行了finalize方法之后仍然没有雨GC Roots有直接或间接的引用,则该对象会被回收;
六、有哪几种垃圾回收器,有哪些优缺点?cms和g1的区别?
垃圾回收器主要分为以下几种
Serial、parNew、parallel Scavenge、 Serial Old、Parallel Old、CMS、G1
Serial
单线程的收集器,收集垃圾时,必须stop the world,使用复制算法
ParNew
Serial收集器的多线程版本,也需要stop the world,使用复制算法
ParallelScavenge
新生代收集器,复制算法的收集器,并发的多线程收集器,目标是打到一个可控的吞吐量,和parNew的最大区别是GC自动调节策略;虚拟机会根据系统的运行状态手机性能监控信息,动态设置这些参数,以提供最优停顿时间和最高的吞吐量;
Serial Old
是ParallelScavenge收集器的老年代版本,使用多线程,标记-整理算法
CMS
是一种以获得最短回收停顿时间为目标的收集器,标记清除算法,运作过程:初始标记、并发标记、重新标记、并发清除,收集结束会产生大量的空间碎片;
G1
标记整理算法实现,运作流程主要包括以下:初始标记。并发标记、最终标记、筛选回收。不会产生空间碎片,可以精确地控制停顿;G1将整个堆分为大小相等的多个区域,G1跟踪每个区域的大小,在后台维护一个优先级列表,每次根据允许的收集时间,优先回收价值最大的区域,已达到在有限时间内获取尽可能高的回收效率
七、什么是类加载?
虚拟机把描述类的数据加载到内存中,并对数据进行校验、解析和初始化,最终变成可以被虚拟机直接使用的class对象。
八、类加载的过程
主要分为以下几个过程:
加载、验证、准备、解析、初始化;
加载:加载分为三步
- 通过类的全限定类名获取该类的二进制流
- 将该二进制流的静态存储结构转为方法区的运行时数据结构;
- 在堆中为该类生成一个class对象
验证
验证该class文件中的字节流信息是否符合虚拟机的要求,不会威胁到jvm的安全
准备
为class对象的静态变量分配内存,初始化其初始值
初始化
到了初始化阶段,才开始执行类中定义的java代码;初始化阶段是调用类构造器的过程;
九、什么是类加载器,常见的类加载器有哪些?
类加载器是指:
通过一个类的全限定性类名获取该类的二进制字节流叫做类加载器;
类加载器分为以下四种
- 启动类加载器:用来加载java核心类库,无法被java程序直接引用;
- 扩展类加载器:用来加载java的扩展库,java的虚拟机实现会提供一个扩展库目录,该类加载器在扩展目录里面查找并加载java类
- 系统类加载器:它根据java的类路径来加载类,一般来说,java应用的类都是通过它来加载的;
- 自定义加载器:由java语言实现,继承自ClassLoader