之前我们在面试的时候是不是会经常被问到相关JVM的问题?来看看下面来自面试官的灵魂 “拷问”
- 请谈谈你对jvm的理解?
- 什么是OOM,什么是栈溢出StackOverFlowError?如何分析?
- JVM常用调优参数有哪些?
- 内存快照如何抓取,怎么分析Dump文件?
- 谈谈JVM中,你对类加载器的认识?
JVM运行在什么地方?
JVM当然是运行在操作系统之上的,可以说是一个Java运行环境(JRE)。
JVM体系结构以及名词介绍
图例:
上图所述:可以看到我们常说的JVM调优,事实上就是在调堆!
名词介绍:
方法区(Method Area)
方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊的方法,如构造函数,接口代码也在此定义,简单说,所有定义的方法信息都保存在该区域。静态变量、常量、类信息(构造方法、接口定义)、运行时的常量池存在方法区中,但是实例存在堆内存中,和方法区无关。
栈(Stack)
是一种数据结构,程序=数据结构+算法。容量小,存取速度快,适合存储生命周期短的数据,栈中存储变量(称为局部变量),特点是先进后出,使用完后就会销毁;存放对象的引用。
堆(Heap)
一个JVM只有一个堆内存,堆内存的大小时可以调节的。类加载器读取了类文件后,一般会把类、方法、常量、变量放到堆中,保存我们所有引用的真实对象。
堆内存中还要细分三个区域:
- 新生代:类诞生成长的地方,甚至死亡。
伊甸园区(Eden):所有的对象都是在伊甸园区new出来的。
幸存者区(from,to):若类在伊甸园区没有被垃圾回收掉(轻GC),则进去幸存者区。- 养老区:若在幸存者区活下来的(重GC),则进去养老区。若此区域在被占满,则报 OutOfMemoryError: Java heap space。
OOM解决:1.调参数,尝试扩大堆内存的运行空间 -Xms1024m -Xmx1024m -XX:+PrintGCDetails 看结果;
2.分析内存,看哪个地方出现问题(JProfiler工具),分析Dumb文件,定位内存泄漏,获得堆中的数据。
-Xms10m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError 下载Dumb文件,用JProfiler打开分析- 永久区(元空间):存放JDK自身携带的Class对象,interface元数据,即Java运行环境。此区域不存在垃圾回收。
JVM调优的一般步骤为:
第1步:分析GC日志及dump文件,判断是否需要优化,确定瓶颈问题点;
第2步:确定JVM调优量化目标;
第3步:确定JVM调优参数(根据历史JVM参数来调整);
第4步:调优一台服务器,对比观察调优前后的差异;
第5步:不断的分析和调整,直到找到合适的JVM参数配置;
第6步:找到最合适的参数,将这些参数应用到所有服务器,并进行后续跟踪。
JVM参数详情
下面只列举其中的几个常用和容易掌握的配置选项,具体的还需大家查阅orcal官网所给出的相关信息
-Xms |
初始堆大小。如:-Xms256m |
-Xmx |
最大堆大小。如:-Xmx512m |
-Xmn |
新生代大小。通常为 Xmx 的 1/3 或 1/4。新生代 = Eden + 2 个 Survivor 空间。实际可用空间为 = Eden + 1 个 Survivor,即 90% |
-Xss |
JDK1.5+ 每个线程堆栈大小为 1M,一般来说如果栈不是很深的话, 1M 是绝对够用了的。 |
-XX:NewRatio |
新生代与老年代的比例,如 –XX:NewRatio=2,则新生代占整个堆空间的1/3,老年代占2/3 |
-XX:SurvivorRatio |
新生代中 Eden 与 Survivor 的比值。默认值为 8。即 Eden 占新生代空间的 8/10,另外两个 Survivor 各占 1/10 |
-XX:PermSize |
永久代(方法区)的初始大小 |
-XX:MaxPermSize |
永久代(方法区)的最大值 |
-XX:+PrintGCDetails |
打印GC 详细信息 |
-XX:+HeapDumpOnOutOfMemoryError |
让虚拟机在发生内存溢出时 Dump 出当前的内存堆转储快照,以便分析用,可结合JProfiler工具详细分析 |
类加载器
当.java文件执行Java c 命令后形成class File文件,装载到类加载器中,So what is a class loader?
作用:加载class文件。
图例:
补充:当某个类加载器假在某个class文件时,它首先会把这个任务向上委派给这个class文件的父类,若没被加载,则这个class文件才会自己加载。
这就是我们常说的 双亲委派机制。
就好比阳哥的一句话:我爸是李刚,有事找我爹!
GC
GC的作用区域:Method和Heap内
JVM在进行GC时,并不是对这三个区域(新生代、幸存区、老年区)统一回收,大部分是在新生代
GC两种分类:轻GC(作用在新生代以及偶尔的幸存区)、重GC(全局)