内存泄漏的几种情况

内存泄漏也称作"存储渗漏",用动态存储分配函数动态开辟的空间,在使用完毕后未释放,结果导致一直占据该内存单元。直到程序结束。(其实说白了就是该内存空间使用完毕之后未回收)即所谓内存泄漏。
内存泄漏形象的比喻是"操作系统可提供给所有进程的存储空间正在被某个进程榨干",最终结果是程序运行时间越长,占用存储空间越来越多,最终用尽全部存储空间,整个系统崩溃。所以"内存泄漏"是从操作系统的角度来看的。这里的存储空间并不是指物理内存,而是指虚拟内存大小,这个虚拟内存大小取决于磁盘交换区设定的大小。由程序申请的一块内存,如果没有任何一个指针指向它,那么这块内存就泄漏了。

while(1){malloc(1024);}这种进程

在Java中内部类的定义与使用一般为成员内部类与匿名内部类,他们的对象都会隐式持有外部类对象的引用,影响外部类对象的回收。

GC只会回收没有被引用或者根集不可到达的对象(取决于GC算法),内部类在生命周期内始终持有外部类的对象的引用,造成外部类的对象始终不满足GC的回收条件,反映在内存上就是内存泄露。(如,Android中Activity的内存泄露)

解决方案为

1.将内部类定义为static

2.用static的变量引用匿名内部类的实例

1.成员内部类
/**
 * 成员内部类
 */
public class Outer {
    private int outer = 1;
 
    class Inner {
        public void show() {
            System.out.println("outer =" + outer);
        }
    }
 
    public static void main(String[] args) {
        Outer.Inner in = new Outer().new Inner();
        in.show();
 
    }
 
}
 
2.方法内部类
/**
 * 方法内部类
 * 
 */
public class Outer2 {
    private int outer = 1;
 
    public void showOut() {
 
        class Inner {
            public void show() {
                System.out.println("outer =" + outer);
            }
        }
        new Inner().show();
    }
 
    public static void main(String[] args) {
        Outer2 in = new Outer2();
        in.showOut();
 
    }
}
防止Handler引起内存泄漏:
方法一:通过程序逻辑进行保护:

(1)在关闭Activity时停掉对应的后台线程。线程停止就相当于切断了Handle和外部链接的线,Activity自然会在合适的时候被回收。

(2)如果Handler是被delay的Message持有了引用,那就使用Handler的removeCallbacks()方法将消息对象从消息队列移除即可。

方法二:将Handler声明为静态类,静态类不持有外部类的对象,所以Activity可以被随意回收。此处使用了弱引用WeakReference,也就是说当在内存不足时,系统会销毁弱/回收引用引用的对象,从而达到优化内存的目的。优化后代码如下:

public abstract class WeakHandler<T> extends Handler {

    protected WeakReference<T> reference;

    //创建子线程Handler使用的构造器
    public WeakHandler(Looper looper, T reference) {
        super(looper);
        this.reference = new WeakReference<>(reference);
    }

    //创建主线程Handler使用的构造器
    public WeakHandler(T reference) {
        this.reference = new WeakReference<>(reference);
    }

    @Override
    public void handleMessage(Message msg) {
        super.handleMessage(msg);
        T t = reference.get();
        if (t == null)
            return;
        handleMessage(t, msg);
    }

    protected abstract void handleMessage(T t, Message message);

}
上述代码,我们使用了泛型,这个泛型就是我们之前说的当前类,同时提供了两种构造器,这样不管我们是创建主线程还是非主线程Handler对象时,都不会造成内存泄漏了。
 

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 在 C/C++ 程序中,内存泄漏通常是由于程序员在使用动态内存分配函数(如 malloc、calloc、realloc)时疏忽或错误导致的。 一种常见的情况是程序员在使用动态内存分配函数申请内存后,忘记调用相应的内存释放函数(如 free)释放内存。如果这种情况发生了多次,那么会导致系统内存的浪费,并可能导致程序崩溃。 另一种情况是,程序员调用了内存释放函数,但是在调用之后仍然继续使用了已经释放的内存区域。这种情况可能会导致程序崩溃或数据错误。 此外,程序员可能会在释放指针时忘记将指针赋值为 NULL,导致内存泄漏。 在 C++ 中,还有另一种情况,即程序员使用了 new 关键字动态分配内存,但忘记使用 delete 关键字释放内存。 总的来说,内存泄漏的原因很多,但大多数情况都是由于程序员在使用动态内存分配函数时疏忽或错误导致的。程序员应该牢记内存管理的原则,尽量减少内存泄漏的发生。 ### 回答2: C/C++中可能导致内存泄漏情况几种: 1. 未释放堆内存:在使用`malloc()`、`new`等函数动态分配内存时,必须使用`free()`、`delete`等函数释放内存。如果忘记释放内存或者释放的次数不正确,就会出现内存泄漏的问题。 2. 循环引用:在使用引用计数的方式管理内存时,如果出现循环引用情况即两个对象相互引用,而没有外部引用指向它们,引用计数就无法减到0,导致内存泄漏。 3. 未关闭文件句柄:在使用文件操作函数`fopen()`、`open()`等打开文件时,需要通过`fclose()`、`close()`等函数来关闭文件句柄。如果忘记关闭文件句柄,系统资源将无法释放,造成内存泄漏。 4. 未释放系统资源:除了内存和文件句柄之外,还有其他系统资源也需要手动释放,如数据库连接、网络连接等。如果在使用完这些资源后没有正确释放它们,就会导致内存泄漏。 5. 堆栈不匹配:在使用C/C++的堆栈内存时,需要确保每次`malloc()`或`new`的内存分配与`free()`或`delete`的内存释放是匹配的,否则会发生内存泄漏。 6. 重复分配内存:如果在已有指针变量上多次调用`malloc()`或`new`,而没有对之前分配的内存进行释放,就会导致内存泄漏。 以上是一些常见的C/C++中导致内存泄漏情况,正确管理内存和资源是保证程序运行稳定性和性能的重要一环,开发者需要注意避免这些问题的发生。 ### 回答3: 在C/C++程序开发中,存在几种常见的情况会导致内存泄漏: 1. 动态内存分配没有被正确释放:如果在程序中使用malloc、new等方法分配内存,但是忘记释放对应的内存,则会造成内存泄漏。例如,如果在一个循环中重复分配内存但没有释放,最终会耗尽系统内存。 2. 对象生命周期没有被正确管理:在C++中,如果对象的析构函数中没有正确释放申请的资源(如内存、文件、数据库连接等),则会导致内存泄漏。这通常发生在没有及时调用对象的析构函数或者程序逻辑错误导致无法调用析构函数的情况。 3. 全局变量未释放:全局变量会在程序运行期间一直存在,如果在全局变量中分配了动态内存但未释放,那么这部分内存会一直被占用而无法回收,导致内存泄漏。因此,在使用全局变量时,需要注意释放对应的资源。 4. 异常情况未被处理:如果程序存在异常情况,但没有正确处理,导致跳过了内存释放的代码段,就会导致内存泄漏。例如,try-catch块内没有对内存进行释放操作。 5. 循环引用导致内存泄漏:在使用动态内存分配时,如果存在循环引用(两个或多个对象相互引用且没有其他对象引用它们),并且没有采用有效的内存释放策略,就会导致内存泄漏。这种情况下需要特别注意对象的析构函数中释放相关的资源。 总之,当程序分配了内存资源但没有及时释放,或者释放不完全时,就会造成内存泄漏。为了避免内存泄漏,需要在程序中正确管理内存的申请和释放,及时释放不再使用的内存
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值