第一个问题,Java在什么时候容易发生内存泄露?
1、静态集合类,如 HashMap、Vector 等,因为GC是不会回收静态类的
2、各种连接,如 数据库连接、网络连接、IO连接等如果没有调用 close 关闭,GC不会回收导致的内存泄露,所以我们一再强调使用完连接之后一定要在 finally 中关闭掉
3、强引用,Object obj = new Object() 这种使用 new 方式创建的就属于强引用,(如果有GCRoots的强引用)垃圾回收器绝对不会回收它,当内存不足时宁愿抛出 OOM 错误使得程序异常停止也不会回收强引用对象。
4、监听器的使用,在释放内存的同时没有相应删除监听器的时候也可能导致内存泄露
第二个问题,什么时候会触发 GC
1、在程序空闲时。
2、程序不可预知的时候/手动调用 System.gc() 。不要手动调用GC,不管是 JVM 自己调用还是手动调用都会使系统陷入短暂的停顿,给垃圾回收让路,手动调用就违背了我们优化 GC 的初衷了
3、堆内存不足时 GC 会被调用。当应用线程在运行,并在运行过程中穿件对象,若这时内存空间不足,JVM就会强制的调用 GC ,一遍回收内存用于新的分配。若GC 一次之后仍不满足内存分配的要求,JVM会再次进行两次 GC 做进一步尝试,若还是不满足需求则会抛出 “out of memort” 的错误使应用异常停止
GC是怎样判断对象存活的
可达性分析(Java中使用)
这个算法的基本思路就是通过一系列的称为 “GC Roots” 的对象作为起始点,从这些节点开始向下搜索,搜索走过的路径称为引用链,当一个对象到GCRoots没有任何引用链相连时,则证明 此对象是不可用的。
作为 GC Roots 的对象包括下面几种:
- 当前虚拟机栈中局部变量表中的引用对象
- 当前本地方法栈中局部变量中的引用的对象
- 方法区中类静态属性引用的对象
- 方法区中常量引用的对象