JAVA 内存泄露详解(原因、例子及解决)

转载请注明出处:http://blog.csdn.net/anxpp/article/details/51325838,谢谢!

Java的一个重要特性就是通过垃圾收集器(GC)自动管理内存的回收,而不需要程序员自己来释放内存。理论上Java中所有不会再被利用的对象所占用的内存,都可以被GC回收,但是Java也存在内存泄露,但它的表现与C++不同。

JAVA 中的内存管理

要了解Java中的内存泄露,首先就得知道Java中的内存是如何管理的。

在Java程序中,我们通常使用new为对象分配内存,而这些内存空间都在堆(Heap)上。

下面看一个示例:

  1. public class Simple {
  2. public static void main(String args[]){
  3. Object object1 = new Object();//obj1
  4. Object object2 = new Object();//obj2
  5. object2 = object1;
  6. //...此时,obj2是可以被清理的
  7. }
  8. }

Java使用有向图的方式进行内存管理:

在有向图中,我们叫作obj1是可达的,obj2就是不可达的,显然不可达的可以被清理。

内存的释放,也即清理那些不可达的对象,是由GC决定和执行的,所以GC会监控每一个对象的状态,包括申请、引用、被引用和赋值等。释放对象的根本原则就是对象不会再被使用

  • 给对象赋予了空值null,之后再没有调用过。
  • 另一个是给对象赋予了新值,这样重新分配了内存空间。

通常,会认为在堆上分配对象的代价比较大,但是GC却优化了这一操作:C++中,在堆上分配一块内存,会查找一块适用的内存加以分配,如果对象销毁,这块内存就可以重用;而Java中,就想一条长的带子,每分配一个新的对象,Java的“堆指针”就向后移动到尚未分配的区域。所以,Java分配内存的效率,可与C++媲美。

但是这种工作方式有一个问题:如果频繁的申请内存,资源将会耗尽。这时GC就介入了进来,它会回收空间,并使堆中的对象排列更紧凑。这样,就始终会有足够大的内存空间可以分配。

gc清理时的引用计数方式:当引用连接至新对象时,引用计数+1;当某个引用离开作用域或被设置为null时,引用计数-1,GC发现这个计数为0时,就回收其占用的内存。这个开销会在引用程序的整个生命周期发生,并且不能处理循环引用的情况。所以这种方式只是用来说明GC的工作方式,而不会被任何一种Java虚拟机应用。

多数GC采用一种自适应的清理方式(加上其他附加的用于提升速度的技术),主要依据是找出任何“活”的对象,然后采用“自适应的、分代的、停止-复制、标记-清理”式的垃圾回收器。具体不介绍太多,这不是本文重点。

JAVA 中的内存泄露

Java中的内存泄露,广义并通俗的说,就是:不再会被使用的对象的内存不能被回收,就是内存泄露。

Java中的内存泄露与C++中的表现有所不同。

在C++中,所有被分配了内存的对象,不再使用后,都必须程序员手动的释放他们。所以,每个类,都会含有一个析构函数,作用就是完成清理工作,如果我们忘记了某些对象的释放,就会造成内存泄露。

但是在Java中,我们不用(也没办法)自己释放内存,无用的对象由GC自动清理,这也极大的简化了我们的编程工作。但,实际有时候一些不再会被使用的对象,在GC看来不能被释放,就会造成内存泄露。

我们知道,对象都是有生命周期的,有的长,有的短,如果长生命周期的对象持有短生命周期的引用,就很可能会出现内存泄露。我们举一个简单的例子:

  1. public class Simple {
  2. Object object;
  3. public void method1(){
  4. object = new Object();
  5. //...其他代码
  6. }
  7. }

这里的object实例,其实我们期望它只作用于method1()方法中,且其他地方不会再用到它,但是,当method1()方法执行完成后,object对象所分配的内存不会马上被认为是可以被释放的对象,只有在Simple类创建的对象被释放后才会被释放,严格的说,这就是一种内存泄露。解决方法就是将object作为method1()方法中的局部变量。当然,如果一定要这么写,可以改为这样:

  1. public class Simple {
  2. Object object;
  3. public void method1(){
  4. object = new Object();
  5. //...其他代码
  6. object = null;
  7. }
  8. }

这样,之前“new Object()”分配的内存,就可以被GC回收。

到这里,Java的内存泄露应该都比较清楚了。下面再进一步说明:

  • 在堆中的分配的内存,在没有将其释放掉的时候,就将所有能访问这块内存的方式都删掉(如指针重新赋值),这是针对c++等语言的,Java中的GC会帮我们处理这种情况,所以我们无需关心。
  • 在内存对象明明已经不需要的时候,还仍然保留着这块内存和它的访问方式(引用),这是所有语言都有可能会出现的内存泄漏方式。编程时如果不小心,我们很容易发生这种情况,如果不太严重,可能就只是短暂的内存泄露。

一些容易发生内存泄露的例子和解决方法

像上面例子中的情况很容易发生,也是我们最容易忽略并引发内存泄露的情况,解决的原则就是尽量减小对象的作用域(比如android studio中,上面的代码就会发出警告,并给出的建议是将类的成员变量改写为方法内的局部变量)以及手动设置null值。

至于作用域,需要在我们编写代码时多注意;null值的手动设置,我们可以看一下Java容器LinkedList源码(可参考:Java之LinkedList源码解读(JDK 1.8))的删除指定节点的内部方法:

  1. //删除指定节点并返回被删除的元素值
  2. E unlink(Node<E> x) {
  3. //获取当前值和前后节点
  4. final E element = x.item;
  5. final Node<E> next = x.next;
  6. final Node<E> prev = x.prev;
  7. if (prev == null) {
  8. first = next; //如果前一个节点为空(如当前节点为首节点),后一个节点成为新的首节点
  9. } else {
  10. prev.next = next;//如果前一个节点不为空,那么他先后指向当前的下一个节点
  11. x.prev = null;
  12. }
  13. if (next == null) {
  14. last = prev; //如果后一个节点为空(如当前节点为尾节点),当前节点前一个成为新的尾节点
  15. } else {
  16. next.prev = prev;//如果后一个节点不为空,后一个节点向前指向当前的前一个节点
  17. x.next = null;
  18. }
  19. x.item = null;
  20. size--;
  21. modCount++;
  22. return element;
  23. }

除了修改节点间的关联关系,我们还要做的就是赋值为null的操作,不管GC何时会开始清理,我们都应及时的将无用的对象标记为可被清理的对象。

我们知道Java容器ArrayList是数组实现的(可参考:Java之ArrayList源码解读(JDK 1.8)),如果我们要为其写一个pop()(弹出)方法,可能会是这样:

  1. public E pop(){
  2. if(size == 0)
  3. return null;
  4. else
  5. return (E) elementData[--size];
  6. }

写法很简洁,但这里却会造成内存溢出:elementData[size-1]依然持有E类型对象的引用,并且暂时不能被GC回收。我们可以如下修改:

  1. public E pop(){
  2. if(size == 0)
  3. return null;
  4. else{
  5. E e = (E) elementData[--size];
  6. elementData[size] = null;
  7. return e;
  8. }
  9. }

我们写代码并不能一味最求简洁,首要是保证其正确性。

容器使用时的内存泄露

在很多文章中可能看到一个如下内存泄露例子:

  1. Vector v = new Vector();
  2. for (int i = 1; i<100; i++)
  3. {
  4. Object o = new Object();
  5. v.add(o);
  6. o = null;
  7. }

可能很多人一开始开始不理解,下面我们将上面的代码完整一下就好理解了:

  1. void method(){
  2. Vector vector = new Vector();
  3. for (int i = 1; i<100; i++)
  4. {
  5. Object object = new Object();
  6. vector.add(object);
  7. object = null;
  8. }
  9. //...对vector的操作
  10. //...与vector无关的其他操作
  11. }

这里内存泄露指的是在对vector操作完成之后,执行下面的代码时,如果发生了GC操作,这一系列的object是没法被回收的,而此处的内存泄露可能是短暂的,因为在整个method()方法执行完成后,那些对象还是可以被回收。这里要解决很简单,手动赋值为null即可:

  1. void method(){
  2. Vector vector = new Vector();
  3. for (int i = 1; i<100; i++)
  4. {
  5. Object object = new Object();
  6. vector.add(object);
  7. object = null;
  8. }
  9. //...对v的操作
  10. vector = null;
  11. //...与v无关的其他操作
  12. }

上面Vector已经过时了,不过只是使用老的例子来做内存泄露的介绍。我们使用容器时很容易发生内存泄露,就如上面的例子,不过上例中,容器时方法内的局部变量,造成的内存泄漏影响可能不算很大(但我们也应该避免),但是,如果这个容器作为一个类的成员变量,甚至是一个静态(static)的成员变量时,就要更加注意内存泄露了。

下面也是一种使用容器时可能会发生的错误:

  1. public class CollectionMemory {
  2. public static void main(String s[]){
  3. Set<MyObject> objects = new LinkedHashSet<MyObject>();
  4. objects.add(new MyObject());
  5. objects.add(new MyObject());
  6. objects.add(new MyObject());
  7. System.out.println(objects.size());
  8. while(true){
  9. objects.add(new MyObject());
  10. }
  11. }
  12. }
  13. class MyObject{
  14. //设置默认数组长度为99999更快的发生OutOfMemoryError
  15. List<String> list = new ArrayList<>(99999);
  16. }

运行上面的代码将很快报错:

  1. 3
  2. Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
  3. at java.util.ArrayList.<init>(ArrayList.java:152)
  4. at com.anxpp.memory.MyObject.<init>(CollectionMemory.java:21)
  5. at com.anxpp.memory.CollectionMemory.main(CollectionMemory.java:16)

如果足够了解Java的容器,上面的错误是不可能发生的。这里也推荐一篇本人介绍Java容器的文章:...

容器Set指存放唯一的元素,是通过对象的equals()方法来比较的,但是Java中所有类都直接或间接继承至Object类,Object类的equals()方法比较的是对象的地址,上例中,就会一直添加元素直到内存溢出。

所以,上例严格的说是容器的错误使用导致的内存溢出。

就Set而言,remove()方法也是通过equals()方法来删除匹配的元素的,如果一个对象确实提供了正确的equals()方法,但是切记不要在修改这个对象后使用remove(),这也可能会发生内存泄露。

各种提供了close()方法的对象

比如数据库连接(dataSourse.getConnection()),网络连接(socket)和io连接,以及使用其他框架的时候,除非其显式的调用了其close()方法(或类似方法)将其连接关闭,否则是不会自动被GC回收的。其实原因依然是长生命周期对象持有短生命周期对象的引用。

可能很多人使用过Hibernate,我们操作数据库时,通过SessionFactory获取一个session:

  1. Session session=sessionFactory.openSession();

完成后我们必须调用close()方法关闭:

  1. session.close();

SessionFactory就是一个长生命周期的对象,而session相对是个短生命周期的对象,而且框架这么设计是合理的:它并不清楚我们要使用session到多久,于是只能提供一个方法让我们自己决定何时不再使用。

因为在close()方法调用之前,可能会抛出异常而导致方法不能被调用,我们通常使用try语言,然后再final语句中执行close()或清理工作:

  1. try{
  2. session=sessionFactory.openSession();
  3. //...其他操作
  4. }finally{
  5. session.close();
  6. }

单例模式导致的内存泄露

单例模式,很多时候我们可以把它的生命周期与整个程序的生命周期看做差不多的,所以是一个长生命周期的对象。如果这个对象持有其他对象的引用,也很容易发生内存泄露。

内部类和外部模块的引用

其实原理依然是一样的,只是出现的方便不一样而已。

与清理相关的方法

本文主要谈论gc()和finalize()方法。

gc()

对于程序员来说,GC基本是透明的,不可见的。运行GC的函数是System.gc(),调用后启动垃圾回收器来回收。

但是根据Java语言规范定义, 该函数不保证JVM的垃圾收集器一定会执行。因为,不同的JVM实现者可能使用不同的算法管理GC。通常,GC的线程的优先级别较低。

JVM调用GC的策略也有很多种,有的是内存使用到达一定程度时,GC才开始工作,也有定时执行的,有的是平缓执行GC,有的是中断式执行GC。但通常来说,我们不需要关心这些。除非在一些特定的场合,GC的执行影响应用程序的性能,例如对于基于Web的实时系统,如网络游戏等,用户不希望GC突然中断应用程序执行而进行垃圾回收,那么我们需要调整GC的参数,让GC能够通过平缓的方式释放内存,例如将垃圾回收分解为一系列的小步骤执行,Sun提供的HotSpot JVM就支持这一特性。

finalize()

finalize()是Object类中的方法。

了解C++的都知道有个析构函数,但是注意,finalize()绝不等于C++中的析构函数。

Java编程思想中是这么解释的:一旦GC准备好释放对象所占用的的存储空间,将先调用其finalize()方法,并在下一次GC回收动作发生时,才会真正回收对象占用的内存,所以一些清理工作,我们可以放到finalize()中。

改方法的一个重要的用途是:当在java中条用非java代码(如c和c++)时,在这些非java代码中可能会用到相应的申请内存的操作(如c的malloc()函数),而在这些非java代码中并没有有效的释放这些内存,就可以使用finalize()方法,并在里面调用本地方法的free()等函数。

所以finalize()并不适合用作普通的清理工作。

不过有时候,该方法也有一定的用处:

如果存在一系列对象,对象中有一个状态为false,如果我们已经处理过这个对象,状态会变为true,为了避免有被遗漏而没有处理的对象,就可以使用finalize()方法:

  1. class MyObject{
  2. boolean state = false;
  3. public void deal(){
  4. //...一些处理操作
  5. state = true;
  6. }
  7. @Override
  8. protected void finalize(){
  9. if(!state){
  10. System.out.println("ERROR:" + "对象未处理!");
  11. }
  12. }
  13. //...
  14. }

但是从很多方面了解,该方法都是被推荐不要使用的,并被认为是多余的。



总的来说,内存泄露问题,还是编码不认真导致的,我们并不能责怪JVM没有更合理的清理。

时间: 2024-10-14 10:37:15

JAVA 内存泄露详解(原因、例子及解决)的相关文章

java内存泄露详解

很多人有疑问,java有很好的垃圾回收机制,怎么会有内存泄露?其实是有的,那么何为内存泄露?在Java中所谓内存泄露就是指在程序运行的过程中产生了一些对象,当不需要这些对象时,他们却没有被垃圾回收掉,而且程序运行中很难发现这个对象,它始终占据着内存却没有发挥作用. 我举这样一个例子,在现实开发中我们需要自定义一个先进后出的栈集合,代码如下: 这个代码看起来和运行起来都没问题,但是,这里有个很隐晦的问题,就是在pop()方法里面,我们首先找到集合最后一个元素的下标,然后按照下标从集合中取出,但是这

JAVA 内存泄露详解

一.Java内存回收机制  不论哪种语言的内存分配方式,都需要返回所分配内存的真实地址,也就是返回一个指针到内存块的首地址.Java中对象是采用new或者反射的方法创建的,这些对象的创建都是在堆(Heap)中分配的,所有对象的回收都是由Java虚拟机通过垃圾回收机制完成的.GC为了能够正确释放对象,会监控每个对象的运行状况,对他们的申请.引用.被引用.赋值等状况进行监控,Java会使用有向图的方法进行管理内存,实时监控对象是否可以达到,如果不可到达,则就将其回收,这样也可以消除引用循环的问题.在

[转]Java内存溢出详解及解决方案

原文地址:http://blog.csdn.net/xianmiao2009/article/details/49254391 内存溢出与数据库锁表的问题,可以说是开发人员的噩梦,一般的程序异常,总是可以知道在什么时候或是在什么操作步骤上出现了异常,而且根据堆栈信息也很容易定位到程序中是某处出现了问题.内存溢出与锁表则不然,一般现象是操作一般时间后系统越来越慢,直到死机,但并不能明确是在什么操作上出现的,发生的时间点也没有规律,查看日志或查看数据库也不能定位出问题的代码. 更严重的是内存溢出与数

java内存空间详解

Java内存分配与管理是Java的核心技术之一,之前我们曾介绍过Java的内存管理与内存泄露以及Java垃圾回收方面的知识,今天我们再次深入Java核心,详细介绍一下Java在内存分配方面的知识.一般Java在内存分配时会涉及到以下区域: ◆寄存器:我们在程序中无法控制 ◆栈:存放基本类型的数据和对象的引用,但对象本身不存放在栈中,而是存放在堆中 ◆堆:存放用new产生的数据 ◆静态域:存放在对象中用static定义的静态成员 ◆常量池:存放常量 ◆非RAM存储:硬盘等永久存储空间 Java内存

java内存模型详解

内存模型 (memory model) 内存模型描述的是程序中各变量(实例域.静态域和数组元素)之间的关系,以及在实际计算机系统中将变量存储到内存和从内存取出变量这样的低层细节. 不同平台间的处理器架构将直接影响内存模型的结构. 在C或C++中, 可以利用不同操作平台下的内存模型来编写并发程序. 但是, 这带给开发人员的是, 更高的学习成本.相比之下, java利用了自身虚拟机的优势, 使内存模型不束缚于具体的处理器架构, 真正实现了跨平台.(针对hotspot jvm, jrockit等不同的

Java 内存区域详解

引言 学习Java也有一段时间了,总感觉有些东西学的不是很精通.例如Java内存区域到底是怎么样的?程序是怎么跑的?对象是怎么存放的?这些都影响了我对自己的代码的熟悉程度. 一 运行时数据区域 Java虚拟机在执行java程序的过程中,会把它所管理的内存划分成若干个不同的数据区域(每当运行一个java程序都会启动一个虚拟机).有一本书叫做<Java虚拟机规范>,讲述了Sun公司对Java虚拟机实现的相关规范,其中讲了虚拟机将所管理的内存分为以下几个部分: 程序计数器 虚拟机栈 本地方法区 堆

Java内存溢出详解

一.常见的Java内存溢出有以下三种: 1. java.lang.OutOfMemoryError: Java heap space ----JVM Heap(堆)溢出JVM在启动的时候会自动设置JVM Heap的值,其初始空间(即-Xms)是物理内存的1/64,最大空间(-Xmx)不可超过物理内存.可以利用JVM提供的-Xmn -Xms -Xmx等选项可进行设置.Heap的大小是Young Generation 和Tenured Generaion 之和.在JVM中如果98%的时间是用于GC,

Java学习1:图解Java内存分析详解(实例)

首先需要明白以下几点: 栈空间(stack),连续的存储空间,遵循后进先出的原则,用于存放局部变量. 堆空间(heap),不连续的空间,用于存放new出的对象,或者说是类的实例. 方法区(method),方法区在堆空间内,用于存放①类的代码信息:②静态变量和方法:③常量池(字符串敞亮等,具有共享机制). Java中除了基本数据类型,其他的均是引用类型,包括类.数组等等. 数据类型的默认值 基本数据类型默认值: 数值型:0 浮点型:0.0 布尔型:false 字符型:\u0000 引用类型:nul

7.Java内存模型详解

https://blog.csdn.net/qq_37141773/article/details/103138476 一.虚拟机 同样的java代码在不同平台生成的机器码肯定是不一样的,因为不同的操作系统底层的硬件指令集是不同的. 同一个java代码在windows上生成的机器码可能是0101.......,在linux上生成的可能是1100......,那么这是怎么实现的呢? 不知道同学们还记不记得,在下载jdk的时候,我们在oracle官网,基于不同的操作系统或者位数版本要下载不同的jdk