JVM笔记:
1.JVM内部结构:方法区、JAVA堆、JAVA栈、本地方法栈、垃圾收集器、PC寄存器、执行引擎、本地方法接口。
PC寄存器:每个线程一个PC寄存器,
指向下一条指令的地址
方法区:保存装载的类信息,通常和永久区(Perm)关联在一起
类型的常量区(字符串常量信息在JDK7后移动到了堆)
字段、方法信息
方法字节码
线程共享的
JAVA堆:new出来的对象都保存在JAVA堆中,
所有线程全局共享JAVA堆
GC的主要工作空间
JAVA栈:线程私有(同步进程)
栈由一系列帧组成(因此JAVA栈也叫作帧栈)
帧保存一个方法的局部变量、操作数栈、常量池指针
每一次方法调用创建一个帧,并压栈
防止内存泄露方法:栈上分配(创建对象不new)
- 小对象,无逃逸情况,可以分配在栈上
- 直接分配在栈中,可以自动回收,减轻GC压力(堆中数据不会自动回收,需要GC来回收,GC不一定会回收(一般是在CPU空闲或空间不足时自动进行垃圾回收))
- 大对象或者逃逸对象无法栈上分配
解释运行:解释执行以解释方式运行字节码,读一句执行依据
编译运行:将字节码变异成机器码,直接执行机器码,运行时编译,运行快
2.GC:垃圾回收
- GC的对象是堆空间和永久区
GC算法:- 引用计数法:对于一个对象A,只要有任何一个对象引用了A,则A的引用计数器就加1,当引用失效时,引用计数器就减1.只要对象A的引用计数器的值为0,则对象A就不可能再被使用。
- 标记清除:
- 标记 通过根节点,标记所有从根节点开始的可达对象,未被标记的对象就是未被引用的垃圾对象
- 清除 清除所有未被标记的对象
- 标记压缩:
- 标记 也是从根节点开始,对所有可达对(存活对象)象做标记
- 压缩 把所有的存活对象压缩到内存的一端,然后,清理边界外的所有空间。
- 复制算法:不适用于存活率高的场合(比标记压缩多了一块内存,空间浪费)
- 将原有的内存空间分为完全相同的两块,每次只是用其中一块
- 在垃圾回收时,将正在使用的内存中的存活对象复制到未使用的块中
之后,清除正在使用的内存块中的所有对象,交换两个内存的角色,完成垃圾回收
分代思想:依据对象的存活周期进行分类
- 新生代 :短命对象
- 老年代 :长命对象
- 少量对象存活,适合复制算法
- 大量对象存活,适合标记清除或标记压缩
- 年轻代的垃圾回收叫 Young GC
- 年老代的垃圾回收叫 Full GC
- 持久代 :用于存放静态类型数据,如 Java Class, Method 等。
持久代对垃圾回收没有显著影响。但是有些应用可能动态生成或调用一些Class,
例如 hibernate CGLib 等,在这种时候往往需要设置一个比较大的持久代空间来
存放这些运行过程中动态增加的类型。
可触及性:
- 可触及的:从根节点可以触及到的这个对象
- 可复活的:一旦所有引用被释放,就是可复活状态,finalize()中可能复活该对象
- 不可触及的:在finalize()后,可能会进入不可触及状态,不可触及的对象不可能复活,可以回收。真正意义上可 以回收的对象
Stop-The-World:
- JAVA中的一种全局暂停的现象
- 所有JAVA代码停止,native代码可以执行,但不能与JVM交互
1.JVM内部结构:方法区、JAVA堆、JAVA栈、本地方法栈、垃圾收集器、PC寄存器、执行引擎、本地方法接口。
PC寄存器:每个线程一个PC寄存器,
指向下一条指令的地址
方法区:保存装载的类信息,通常和永久区(Perm)关联在一起
类型的常量区(字符串常量信息在JDK7后移动到了堆)
字段、方法信息
方法字节码
线程共享的
JAVA堆:new出来的对象都保存在JAVA堆中,
所有线程全局共享JAVA堆
GC的主要工作空间
JAVA栈:线程私有(同步进程)
栈由一系列帧组成(因此JAVA栈也叫作帧栈)
帧保存一个方法的局部变量、操作数栈、常量池指针
每一次方法调用创建一个帧,并压栈
防止内存泄露方法:栈上分配(创建对象不new)
- 小对象,无逃逸情况,可以分配在栈上
- 直接分配在栈中,可以自动回收,减轻GC压力(堆中数据不会自动回收,需要GC来回收,GC不一定会回收(一般是在CPU空闲或空间不足时自动进行垃圾回收))
- 大对象或者逃逸对象无法栈上分配
解释运行:解释执行以解释方式运行字节码,读一句执行依据
编译运行:将字节码变异成机器码,直接执行机器码,运行时编译,运行快
2.GC:垃圾回收
- GC的对象是堆空间和永久区
GC算法:- 引用计数法:对于一个对象A,只要有任何一个对象引用了A,则A的引用计数器就加1,当引用失效时,引用计数器就减1.只要对象A的引用计数器的值为0,则对象A就不可能再被使用。
- 标记清除:
- 标记 通过根节点,标记所有从根节点开始的可达对象,未被标记的对象就是未被引用的垃圾对象
- 清除 清除所有未被标记的对象
- 标记压缩:
- 标记 也是从根节点开始,对所有可达对(存活对象)象做标记
- 压缩 把所有的存活对象压缩到内存的一端,然后,清理边界外的所有空间。
- 复制算法:不适用于存活率高的场合(比标记压缩多了一块内存,空间浪费)
- 将原有的内存空间分为完全相同的两块,每次只是用其中一块
- 在垃圾回收时,将正在使用的内存中的存活对象复制到未使用的块中
之后,清除正在使用的内存块中的所有对象,交换两个内存的角色,完成垃圾回收
分代思想:依据对象的存活周期进行分类
- 新生代 :短命对象
- 老年代 :长命对象
- 少量对象存活,适合复制算法
- 大量对象存活,适合标记清除或标记压缩
- 年轻代的垃圾回收叫 Young GC
- 年老代的垃圾回收叫 Full GC
- 持久代 :用于存放静态类型数据,如 Java Class, Method 等。
持久代对垃圾回收没有显著影响。但是有些应用可能动态生成或调用一些Class,
例如 hibernate CGLib 等,在这种时候往往需要设置一个比较大的持久代空间来
存放这些运行过程中动态增加的类型。
可触及性:
- 可触及的:从根节点可以触及到的这个对象
- 可复活的:一旦所有引用被释放,就是可复活状态,finalize()中可能复活该对象
- 不可触及的:在finalize()后,可能会进入不可触及状态,不可触及的对象不可能复活,可以回收。真正意义上可 以回收的对象
Stop-The-World:
- JAVA中的一种全局暂停的现象
- 所有JAVA代码停止,native代码可以执行,但不能与JVM交互
- 多半由于GC引起
思维导图来自网络: