1 java为什么需要自动垃圾回收
如果Java中没有自动垃圾回收机制,则通常会出现两种问题
a 忘记回收
忘记回收就会出现内存泄漏,最后产生内存溢出
b 多次回收
多次回收,在一些条件下面对象引用的空间已经回收了,并且这块空间被新的对象引用,但是原对象引用依旧指向这个地址,就会差生误删数据
2 什么是垃圾
a 没有任何引用指向它的对象
b 没有外部指向的多个对象
3 怎么定位垃圾
a refeence count(引用计数)
对象持有被引用的个数,没增加一个引用,计数加1,反之,减1,但引用计数为0的时候表示没有任何引用指向这个对象了
缺点(无法解决多个对象循环被引用,并且这几个对象没有被外部引用)
b GC roots(根可达算法)
gc roots包括
被系统加载器或者根加载器加载的class类,部分自定义加载器的class
处于激活中的线程
方法栈中的对象
本地方法栈中的对象
本地方法栈中的全局变量
处于正在使用中的锁对象
JVM自身持有的对象
不属于GC root的或者是从gc root上不可以到达的对象都属于垃圾
4 常见的垃圾回收算法
a 标记清除算法
从内存中,把属于垃圾的对象,全部标记上,再然后把标记的对象的内存全部回收
缺点:会差生不连续的垃圾碎片
b 复制拷贝算法
再内存使用上,就把内存分成两份,每次创建对象都在其中的一份上,回收的时候,把不属于垃圾的拷贝到另外一份上,然后再把这份内存全部回收
缺点:需要耗费内存
c 标记压缩算法
把不是垃圾的对象移动到连续的内存上,整理
缺点:时效慢,每移动整理对象的时候都要做同步处理
5 JVM内存分代模型(用于分代垃圾回收算法)
部分垃圾回收器使用的模型,比如G1、ZGC垃圾回收器就没有区分新生代,老生代
会产生垃圾回收的地方
新生代+老生代+永久代(jdk1.7)/元数据区(jdk1.8)
永久代/元数据区一般存放 Class
永久代与元数据区的区别,永久代必须要设置大小,(当应用有很多的代理的时候,就会产生很多的Class对象,会把永久代撑爆,出现内存溢出)元数据区可以设置也可以不设置,无限制(受限于物理内存)
永久带存放字符串常量,元数据区不存在,把这块交给了堆,元数据区的回收不属于JVM管理,由系统控制
一般新生代:老生代 = 1 : 3
新生代=Eden+2个suvivor区
Eden : s0 :s1 = 8 : 1 : 1
新生代中的对象一般默认存活率很少,有90%的都会被回收
a 再进行过一个YGC后,大部分对象被回收,剩下的活着对象进入s0
b 再次YGC,把Eden和s0区活着的对象放入s1区
c 再次YGC,把Eden和s1区活着的对象放入s0区
什么时候对象进入老生代
a 年龄足够的情况下,对象进入老生代,新生代中的对象每经过一次YGC,年龄就加1,一般默认的年龄是15,CMS默认年龄6
b Eden区进行YGC,活着的对象太多了,s区装不下了,就进入老生代
c 在创建对象的时候,对象很大,Eden区分配不出来,也进入老生代
当老生代满了的时候就会进行Full Gc
一般我们gc调优的目的就是尽量减少Full Gc次数
6 常见的垃圾回收器
a Serial a stop the world .copying collector which uses a single GC thread(新生代,串行回收)
b Parallel Scavenge a stop the world which uses multiple GC thread(新生代,并行回收)
c ParNew(新生代,配合CMS并行回收)
d Serial Old(老生代)
e Parallel Old (老生代)
f CMS (老生代,并发的,垃圾回收和应用程序同时进行)
g G1
h ZGC(zero gc)
jdk8默认的垃圾回收器 Parallel Scavenge + Parallel Old
第一次写,希望大佬们多多指教