java是否存在内存泄露问题

今天我们来讲讲java内存泄露问题,内存泄露是指一个不在被程序使用的对象或变量还在内存中占有存储空间,在C/C++语言中,内存的分配与释放是由开发人员负责的,如果开发人员忘记释放已分配的内存就会造成内存泄露,而在java语言中引进了垃圾回收机制,由垃圾回收器负责回收不在使用的对象,那么,我们想一个问题,既然有垃圾回收器开负责回收垃圾,那么还是否存在内存泄露的问题了?

   其实,在java语言中,判断一个内存空间是否符合垃圾回收的标准有两个,第一,给对象赋予了空置null,以后在没有被使用过,第二,给对象赋予了新值,重新分配了内存空间,一般来讲,内存泄露主要有两种情况,一是在堆中申请的空间没有被释放,二是对象已不再被使用,但还任然在内存中保留着,垃圾回收机制的引入可以有效的解决第一种情况,而对于第二种情况,垃圾回收机制则无法保证不再使用的对象会被释放,因此,java语言中的内存泄露主要指的是第二种情况。

我们来举个栗子来介绍下java语言中的内存泄露

public class TestDemo  {
    public static void main(String[] args) {
        Vector v = new Vector(10);
        for (int i = 0; i < 10; i++) {
         Object o = new Object();
         v.add(o);
        }
    }
}

在上述列子的循环中,不断创建新的对象加到Vector对象中,当退出循环后,“o”的作用域将会结束,但是由于v在使用这些对象,因此垃圾回收器无法将其回收,此时就造成了内存泄露,只有将这些对象从Vector中删除才能释放创建的这些对象。

在java语言中,容易引起内存泄露的原因有很多,主要由以下几个方面的内容。

(1)静态集合类。例如HashMap和Vector.如果这些容器为静态的,由于它们的生命周期与程序一致,那么容器中的对象在程序结束之前将不能被释放,从而造成内存泄露,如上列所示。

(2)各种连接,列如数据库连接,网络连接以及io连接等,在对数据库进行操作的过程中,首先需要建立与数据库的连接,当不在使用时,需要调用close方法来释放与数据库的连接,只有连接被关闭后,垃圾回收器才会回收对应的对象,否则,如果在访问数据库的过程中,对Connection,Statement或ResultSet不显示的关闭,将会造成大量的对象无法被回收,从而引起内存泄露。

(3)监听器。在java语言中,往往会使用到监听器,通常一个应用中会用到多个监听器,但在释放对象的同时,往往没有相应得删除监听器,这也可能导致内存泄露。

(4)变量不合理的作用域,一般而言,如果一个变量定义的作用范围大于其使用范围,很有可能会造成内存泄露,另一方面,如果没有及时地把对象设置为null,很有可能会导致内存泄露的发生。

我们来举个例子:

  private String msg;
        public void recieveMsg(){
            readFromNet();//从网络接收数据保存到msg中国
            saveDB();//把msg保存到数据库中
        }

在上述伪代码中,通过readFromNet()方法接收的消息保存在变量msg中,然后调用saveDB()方法把msg的内容保存到数据库中,此时msg已经没用了,但是由于msg的生命周期与对象的生命周期相同,此时msg还不能被回收,因此造成了内存泄露,对于这个问题,有如下两种解决方法,第一种方法,由于msg作用范围只在recieveMsg()方法内,因此可以把msg定义为这个方法的局部变量,当方法结束后,msg的生命周期就会结束。此时垃圾回收器就可以回收msg的内容了,第二种方法,在使用完msg后就把msg设置为null,这样垃圾回收器也会自动回收msg内容所占的内存空间。

(5)单列模式可能会造成内存泄露,单列模式的实现方法有很多种,下列中所使用的单列模式就可能会造成内存泄露。

class BigClass {

}
class Singleton{
    private BigClass bc;

    private static Singleton instance = new Singleton(new BigClass());
    private Singleton(BigClass bc){
        this.bc = bc;
    }
    public Singleton getInstance(){
         return instance;
    }
}

在上述实现的单列模式中,Singleton存在一个对对象BigClss的引用,由于单列对象以静态变量的方式存储,因此它在JVM的整个生命周期中都存在,同时由于它有一个对对象Big-Class的引用,这样会导致BigClass类的对象不能够被回收。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值