内存模型
想要知道jvm的垃圾回收首先要知道jvm的内存模型,jvm内存模型包括:程序计数器、堆、方法区、栈,其中栈还分为本地方法栈以及虚拟机栈。
1、程序计数器:程序计数器是用于存放下一条指令所在单元的地址的地方。通俗的讲就是告诉程序下一条代码执行哪一行的。
2、堆:堆主要存放所有类的实例和数组,也是jvm在垃圾回收的时候的主要区域。
3、栈分为虚拟机栈和本地方法栈。虚拟机栈主要储存方法、方法中的局部变量,本地方法栈也是,但是不同的是它储存的是native修饰的。
4、方法区存储类加载的相关信息,以及静态变量、常量、运行时常量池等
垃圾回收
简单讲过内存模型后就可以更加轻松的了解JVM的垃圾回收。分为两个步骤,首先你需要知道什么是垃圾,接下来就是将垃圾清理掉。
垃圾判断的算法:
- 引用计数算法:这个算法比较好理解,就是给每个对象一个记数,如果被引用就加1,如果引用消失就减1。但是这种算法的缺点很明显,没办法解决循环引用的问题,这个严重的问题导致JVM并没有使用这一算法。
- 可达性算法:可达性算法类似树根一样,最上面是GCRoots,所有有用的对象都会直接或者间接的和最上面的GCRoots有着引用链链接。当一个对象没有办法达到,也就是没有任何路径从GCRoots链接到这个对象。GCRoots指代:虚拟机栈本地变量的引用,本地方法栈中的引用,方法区中常量和静态属性的引用。
垃圾回收算法:
- 标记清楚算法:这种算法就是直接遍历所有对象,并且在GC时候需要暂停程序,性能较差。并且通过这种方法回收的内存碎片化比较严重,都是断断续续的
- 标记整理算法:这种算法就是对上面的算法的一个优化。在清楚时并不是像上面那样只清除,而是边清除边将存活对象移动到内存的一端。
- 复制算法:复制算法核心就是将内存分成两块,每次只使用其中一块。当要进行垃圾回收时,将使用的那一块内存中的正在使用对象复制到未使用的内存块中,然后清除原来使用的内存块。优点就是在需要清理对象比较多的情况下效率比较高且没有内存碎片化。缺点就是内存的使用率比较低。
- 分代收集算法:在java8时,堆被分为了两份:新生代和老年代【1:2】,在java7时,还存在一个永久代。对于新生代,内部又被分为了三个区域。Eden区,SO区,S1区【8:1:1】。1)当创建一个对象的时候,那么这个对象会被分配在新生代的Eden区。当Eden 区要满了时候,触发YoungGC。 2)当进行YoungGC后,此时在Eden区存活的对象被移动到SO区,并且当前对象 的年龄会加1,清空Eden区。 3)当再一次触发YoungGC的时候,会把Eden区中存活下来的对象和S0中的对象,移动到S1区中,这些对象的年龄会加1,清空Eden区和SO区。 4)当再一次触发YoungGC的时候,会把Eden区中存活下来的对象和S1中的对 象,移动到SO区中,这些对象的年龄会加1,清空Eden区和S1区。当年龄到某一个限定值(默认是15)时对象会进去老年区,或者当某个年龄的对象加起来的和超过了一半,则大于这个年龄的对象可以进去到老年区。
这篇文章比较枯燥,没有代码也没有图片。都是最近看的JVM整理得来,我觉得相对来说比大多数讲的通俗简易,文章仅代表个人观点!