深入了解JVM

堆的内存划分

堆分为新生代和老年代,新生代占三分之一,老年代占三分之二

新生代又分成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的处理方式是用一个填充对象将此填满,不填的话堆在线性遍历时遇到此空洞就会断

相关推荐
程序员二叉7 小时前
【JUC】线程池全套深度详解|参数|流程|拒绝策略|调优|异常处理
java·开发语言·jvm·算法·面试·juc
小马爱打代码11 小时前
面试题:内存模型与垃圾回收深度解析
jvm
cfm_291414 小时前
JVM底层源码深度解析:读写屏障(Read/Write Barrier)
jvm
wuminyu15 小时前
Java世界中StringTable源码剖析
java·linux·c语言·jvm·c++
醉颜凉16 小时前
Elasticsearch性能优化:JVM GC调优全攻略,彻底解决集群卡顿、吞吐量下降问题
jvm·elasticsearch·性能优化
顺风尿一寸18 小时前
从 Java 到内核:探秘线程改名的完整路径
jvm
lihao lihao20 小时前
linux线程
java·开发语言·jvm
南极企鹅1 天前
JVM-编译执行过程
jvm
苏克贝塔2 天前
.NET开发之.net framework对比.net core
jvm
cfm_29142 天前
JVM垃圾收集算法与收集器深度解析
jvm·测试工具·算法·性能优化