STL类指针容器container<class*> 元素clear 注意事项

在使用 STL 容器时, 涉及到 C++多态(polymorphism)和虚函数时,经常会遇到使用容器存储类指针,当你要删除某个元素或者(erase)或者清空容器(clear)时,直接调用 erase或 clear 函数,指针指向的内存并不会释放,也就是说,对象不会被析构,这时就会导致内存泄漏(memory leak)。
示例代码:

#include <iostream>
#include <list>

using std::cout;


class Base {
public:
    virtual ~Base() {}
};

class Derived : public Base {};


int main(int argc, char* argv[])
{
    using std::list;
    list<Base*> ls;
    ls.push_back(new Base());
    ls.push_back(new Derived());
    ls.push_back(new Base());
    ls.clear();
    return 0;
}

编译后,使用内存检测工具 valgrind运行内存泄漏检测,得:
这里写图片描述
从图中可以发现程序中有三处内存泄漏,正对应着:

ls.push_back(new Base());
ls.push_back(new Derived());
ls.push_back(new Base());

可见,我们直接调用 clear()后指针对应的内存并没有被释放。而要消除内存泄漏,我们必须添加清理代码,对程序做以下修改:

#include <iostream>
#include <list>

using std::cout;


class Base {
public:
    virtual ~Base() {}
};

class Derived : public Base {};


int main(int argc, char* argv[])
{
    using std::list;
    list<Base*> ls;
    ls.push_back(new Base());
    ls.push_back(new Derived());
    ls.push_back(new Base());

    //添加清理代码
    for(list<Base*>::iterator it=ls.begin(); it != ls.end(); ++it)
        delete *it;

    ls.clear();
    return 0;
}

再次运行 valgrind, 我们可以发现原来的内存泄漏消失。
这里写图片描述

有关 valgrind 的使用网上有很多,也可以查看本地帮助, 具体命令

valgrind --help

这里只是简单地开启 –leak-check 选项,另外程序在编译时加入“-g”选项,否则看不到发生内存泄漏的代码位置。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值