Chinaunix首页 | 论坛 | 博客
  • 博客访问: 9472857
  • 博文数量: 1751
  • 博客积分: 12961
  • 博客等级: 上将
  • 技术积分: 20101
  • 用 户 组: 普通用户
  • 注册时间: 2009-01-09 11:25
个人简介

偷得浮生半桶水(半日闲), 好记性不如抄下来(烂笔头). 信息爆炸的时代, 学习是一项持续的工作.

文章分类

全部博文(1751)

文章存档

2024年(27)

2023年(26)

2022年(112)

2021年(217)

2020年(157)

2019年(192)

2018年(81)

2017年(78)

2016年(70)

2015年(52)

2014年(40)

2013年(51)

2012年(85)

2011年(45)

2010年(231)

2009年(287)

分类: Windows平台

2019-07-08 15:06:19

STL中各个容器的内存的释放



  1. 1.vector总是不会释放内存
  2. 删除向量容器的元素时,并不会使空闲的空间被释放,这时可以使用下面的语句达到释放多余空间的目的( s 表示目的容器,T表示容器的元素类型):
  3. vector<T>(s.begin(), s.end(), swap(s));
  4. 即首先用 s 的内容创建一个临时的向量容器对象,再将该容器和 s 交换,这时 s 原先占有的空间已经属于临时对象,该语句执行完成后临时对象会被析构,空间被释放。 

  5. 2.deque有时会释放内存
  6. deque的内存区块不再被使用时,会被释放。deque的内存大小是可缩减的。不过,是不是这么做以及怎么做由实际版本定义。

  7. 3.list,set,multiset,map,multimap总是会释放内存
  8. list一般实现为一个双向链表,而set、multiset、map、multimap一般实现为一个平衡二叉树。当容器中元素被删除时,元素所占的内存被释放。

容器删除的坑.

对于关联容器(如map, set, multimap,multiset),删除当前的iterator,仅仅会使当前的iterator失效,只要在erase时,递增当前iterator即可。这是因为map之类的容器,使用了红黑树来实现,插入、删除一个结点不会对其他结点造成影响
  1. for (iter = container.begin(); it != container.end();)
  2. {
  3.   (*iter)->doSomething();
  4.   if (shouldDelete(*iter))
  5.     container.erase(iter++);
  6.   else
  7.     ++iter;
  8. }

===========================================================
对于序列式容器(如vector,deque),删除当前的iterator会使后面所有元素的iterator都失效。这是因为vetor,deque使用了连续分配的内存,删除一个元素导致后面所有的元素会向前移动一个位置。还好erase方法返回的是下一个有效的iterator。
  1. for (iter = container.begin(); iter != container.end();)
  2. {
  3.   (*it)->doSomething();
  4.   if (shouldDelete(*iter))
  5.     iter = container.erase(iter);
  6.   else
  7.     ++iter;
  8. }
==========================================================
对于list来说,它使用了不连续分配的内存,并且它的erase方法也会返回下一个有效的iterator,因此上面两种方法都可以使用。

******************************************************************************
******************************************************************************
******************************************************************************

其他技巧

删除值为val的元素. 最标准的在vector中删除值为5的元素的方法
void EraseVec()
{
    std::vector vec{ 1, 2, 3, 4, 5, 5, 6, 7 };
    vec.erase(std::remove(vec.begin(), vec.end(), 5), vec.end());
}
remove做的操作是移除而erase是删除. remove 仅仅是把移除的内容放到了相应的内存末尾.

iterator erase (iterator begin, iterator end);
//tips : 删除容器中迭代器在[first; last)范围内的元素
//return : 最后一个被删除的元素的下一个元素的迭代器

iterator remove(iterator begin, iterator end, val);
//tips : 移除容器中迭代器在[first; last)范围内的值为val的元素
//return : 被成功移除的之后,剩余的有效元素的下一个元素的迭代器

删除满足某个条件的元素
void EraseVecIf()
{
    std::vector vec{ 1, 2, 3, 4, 5, 5, 6, 7 }; 
    vec.erase(std::remove_if(vec.begin(), vec.end(), 
    [](int elem)
    {
        return elem % 2 == 0;
    }), 
    vec.end());
}

void EraseList()
{
    std::list ls{ 1, 2, 3, 4, 5, 5, 6, 7 };
    ls.remove(5);
    ls.remove_if([](int elem) { return elem % 2 == 0; });
}




阅读(5384) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~