内存泄漏可能由哪些原因导致呢?

内存泄漏可能由哪些原因导致呢?
内存泄漏可能的原因有很多种:
静态集合类引起内存泄漏
静态集合的生命周期和 JVM 一致,所以静态集合引用的对象不能被释放。
public class OOM {
static List list = new ArrayList();
public void oomTests(){
Object obj = new Object();
list.add(obj);
}
}
单例模式
和上面的例子原理类似,单例对象在初始化后会以静态变量的方式在 JVM 的整个生命周期中存在。如果单例对象持有外部的引用,那么这个外部对象将不能被 GC 回收,导致内存泄漏。
数据连接、IO、Socket 等连接
创建的连接不再使用时,需要调用 close 方法关闭连接,只有连接被关闭后,GC 才会回收对应的对象(Connection,Statement,ResultSet,Session)。忘记关闭这些资源会导致持续占有内存,无法被 GC 回收。
try {
Connection conn = null;
Class.forName("com.mysql.jdbc.Driver");
conn = DriverManager.getConnection("url", "", "");
Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("....");
} catch (Exception e) {
}finally {
//不关闭连接
}
变量不合理的作用域
一个变量的定义作用域大于其使用范围,很可能存在内存泄漏;或不再使用对象没有及时将对象设置为 null,很可能导致内存泄漏的发生。
public class Simple {
Object object;
public void method1(){
object = new Object();
//...其他代码
//由于作用域原因,method1执行完成之后,object 对象所分配的内存不会马上释放
object = null;
}
}
哈希值发生变化
当对象的哈希值发生变化时,如果该对象被存储在使用哈希值进行索引的容器(如HashMap、HashSet等)中,就会导致无法正确找到该对象。因为容器使用哈希值确定对象在内部数据结构中的位置,如果哈希值改变,就无法准确地定位到对象,从而导致内存泄漏。这也是为什么在设计String类时将其设置为不可变类型,以保持其哈希值的稳定性。
ThreadLocal使用不当
使用ThreadLocal时,如果不正确地清除已使用的ThreadLocal变量,就可能导致内存泄漏。这是因为ThreadLocal使用了弱引用来引用线程本地变量,如果在使用完毕后没有及时调用remove方法清除对应的ThreadLocal变量,就有可能导致线程无法被垃圾回收,从而造成内存泄漏。因此,在使用完ThreadLocal后,务必记得调用remove方法进行清除操作,以避免内存泄漏的发生。