JVM学习笔记
JVM
注意:
0、JVM运行时数据区和JVM内存模型不要搞混
1、运行时数据区可以分为:堆、方法区、虚拟机栈、本地方法栈、程序计数器
2、内存模型可以分为:新生代(新生代还可以分为Eden区、Survivor区)、老年代、永久代(java8以后没有永久代,即没有方法区)
3、每一个线程都会有一个虚拟机栈、本地方法栈、程序计数器
JMM.png 运行时数据区.png
JVM的内存划分(JVM运行时数据区):
1、堆、方法区(线程共享数据区)
2、虚拟机栈、本地方法栈、程序计数器(线程隔离数据区)
程序计数器:
0、指向当前线程正在执行的字节码地址 行号
1、java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的。
2、线程私有的,为了切换线程能回到准确的位置
3、执行java方法,记录的是字节码指令地址
4、执行本地方法,计数器值为空
5、不会出现内存溢出
java虚拟机栈
1、线程私有的,创建线程的时候就会创建一个虚拟机栈
2、执行java程序的时候,每个方法都会创建一个栈帧(栈帧分为几个区域:局部变量表、操作数栈、动态连接、方法出口)
3、平时说的变量存放在栈中,其实是存放在栈帧中的局部变量表中,如果是引用类型,只存储对象的引用地址
4、并发量大的时候,会创建若干个虚拟机栈、可能会导致内存溢出(OOM),可以通过吧栈内存调小,提高系统并发量
5、栈的内存调小以后,每个方法会创建一个栈帧,如果栈里面存放大量的栈帧,就会导致栈内存溢出(StackOverFlowError)
本地方法栈
1、本地方法:是非java语言编写的方法,例如,java调用C语言来操作硬件信息
2、线程私有
3、线程调用本地方法是,存储本地方法的局部变量表,本地方法的操作数栈等信息
堆
1、线程共享、存放的都是对象的实例(new 出来的对象都放在堆里)
2、垃圾回收,主要回收堆区
3、为了提升垃圾回收性能,又把堆分为两个区域:新生代和老年代(因为对象的生命周期不一样,所以要分代)
4、新生代又可以划分为三个区域:Eden区、两个Survivor区(From Survivor 和To Sruvivor)
5、Eden区:新建的对象放在Eden区
6、Survivor区:保存新生代GC后还存活的对象
7、Eden:From Survivor:To Survivor = 8:1:1
8、老年代里面的对象存活时间比较长,经过多次的新生代的垃圾收集(默认15次)会进入老年代
9、当堆中的对象实例过多,且大部分都还在使用,会出现内存溢出OOM
方法区
1、线程共享
2、存放已被虚拟机加载的类信息、常量、静态变量
3、习惯被称为永久代
4、永久代也会被回收,主要对常量池的回收、类型的卸载(比如反射生成的大量临时使用的Class信息)
5、当方法区满的时候会出现内存溢出OOM
5、java8以后没有方法区
直接内存
1、不是虚拟机运行时数据区的一部分
2、也不是java虚拟机规范中定义的内存区域