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自身携带的
【一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义】
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文件。