堆的内存划分

堆分为新生代和老年代,新生代占三分之一,老年代占三分之二
新生代又分成Eden和两个Survivor两个区,比例为8:1:1
新对象优先在Eden区分配,满了就会触发Minor GC,存活的放到幸存区,两个幸存区轮流用
每熬过一次GC年龄加1,年龄到15,或者幸存区放不下了就晋升到老年代
老年代满了就触发FULL GC
为什么新生代默认用8:1:1的比例
因为根据统计新生代中的90%以上的对象活不过第一次GC,意味着每次Minor GC只需要预留10%给存活对象就够了,这个比例就是在空间利用率和晋升老年代的风险之间取得平衡,可以通过
-XX:SurvivorRatio 调整
为什么要熬过15次GC
不是必须15次,而是最多15次,jvm对象头中用于记录分代年龄的字段只有4个bit位,这个参数可以进行配置--XX:MaxTenuringThreshold
说说一下JVM类加载器分类与核心功能
类加载器是JVM用来把.Class文件加载到内存,转化成Class对象的组件
java里有如下几类加载器
- 启动类加载器:负责加载<JAVA_HOME>/lib下的核心库如rt.jar
- 扩展类加载器(JDK9之后改为平台类加载器):负责加载<JAVA_HOME>/lib/ext下的扩展库
- 应用程序类加载器:负责加载ClassPath下的类库,包括业务代码和第三方JAR包
- 自定义加载器:负责加载用户自定义路径下的类包,通过继承ClassLoader类并重写findClass方法可以自定义类加载器,典型场景比如Tomcat的WebAppClassLoader,OSGi的模块隔离
Java 中的强引用、软引用、弱引用和虚引用分别是什么?
- 强引用: Object obj =new object()就是强引用,只要存在,GC永远不会回收,即使OOM
- 软引用:通过SoftReference包装的引用,内存够用的时候不会受,内存吃紧快要OOM是才回收
- 弱引用:通过WeakReference包装的引用,只要GC那么就会回收
- 虚引用:唯一的用途是监控对象什么时候被回收
JVM的TLAB是什么?
TLAB是JVM给每个线程划分的一小块私有堆内存,专门用于加速对象分配,不用和别的线程枪锁,提高了分配效率。线程分配时优先从子的TLAB拿内存,不够的话从Eden区申请新的,如果对象太大就直接在Eden区装配,绕过TLAB。
存在内存碎片化问题
比如生了8字节,但下一个对象要16字节,这时候得申请新的TLAB,那老的TLAB里的8字节就浪费了
HotSpot的处理方式是用一个填充对象将此填满,不填的话堆在线性遍历时遇到此空洞就会断