最近在看《Effective STL》【条款17:使用交换技巧修整过剩容量】中提到容器的成函数void swap(container& from),即实现容器对象与from对象的交换。

另外,对于连续内存容器vector和string,还有shrink to fit(收缩到合适,根据容器的实际size设置capacity,减少实际内存分配)的功能。

以vector为例,首先说下,size和capacity、resize和reserve的含义及区别。

1. sizecapacityresizereserve

size:对象的元素个数,除reserve外的几乎所有vector操作都可以影响size;

capacity:容器预留容量,即占用真实内存量,capacity >= size。push_back、insert、resize、reserve操作可以影响capacity,其特点为只增不减(即使erase和clear操作);

resize:可以改变size,如果resize的元素个数大于capacity,则同时改变capacity,增加对象内存空间;

reserve:不改变size,如果reserve的元素个数大于capacity,则改变capacity,增加对象内存空间,反之,则不影响capacity;

代码示例如下

 #include <iostream>
#include <vector> using namespace std; int main()
{
vector<int> v;
cout << "initial: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; v.insert(v.end(), , );
cout << "after insert: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; v.erase(v.end()-, v.end());
cout << "after erase: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; v.reserve();
cout << "after reserve: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; v.resize();
cout << "after resize: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; v.reserve();
cout << "after reserve: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; v.clear();
cout << "after clear: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; return ;
}

输出结果为:

initial:
size of v is
capacity of v is after insert:
size of v is
capacity of v is after erase:
size of v is
capacity of v is after reserve:
size of v is
capacity of v is after resize:
size of v is
capacity of v is after reserve:
size of v is
capacity of v is after clear:
size of v is
capacity of v is

根据上面几个概念及Demo,可以看出来,vector对象的实际内存(即capacity)在多次insert和erase操作后大于实际元素个数(size),即使在clear操作后,capacity依然不会减小为零。

所以,有必要对这种基于内存池的容器做内存收缩。

2. swapshrink to fit

修改vector对象多余内存空间的方法:

vector<int>(v).swap(v); // v为一个vector对象

vector<int>(v)基于v拷贝构造了一个临时vector,其分配了v.size()个元素的内存空间,即capacity为v.size(),然后vector<int>(v)与v交换数据,交换后,v的capacity也减少为v.size(),而临时vector对象vector<int>(v)被销毁。

另外,vector<int>().swap(v); // 可以清除v的内存空间

代码示例如下

 #include <iostream>
#include <vector> using namespace std; int main()
{
vector<int> v(, );
cout << "initial: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; v.erase(v.end()-, v.end());
cout << "after erase: " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; vector<int>(v).swap(v);
cout << "after swap (shrink to fit): " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; vector<int>().swap(v);
cout << "after swap (clear): " << endl;
cout << "size of v is " << v.size() << endl;
cout << "capacity of v is " << v.capacity() << endl << endl; return ;
}

输出结果为:

initial:
size of v is
capacity of v is after erase:
size of v is
capacity of v is after swap (shrink to fit):
size of v is
capacity of v is after swap (clear):
size of v is
capacity of v is

如果使用C++11及更新标准的话,可以直接使用shrink_to_fit函数。

[CPP - STL] swap技巧的更多相关文章

  1. 利用swap技巧去除容器多余的容量

    假设我们预先为容器添加了一部分元素,接着用clear将它们删除,容器内部分配的存储空间实际上不会减小,改变的只是能够访问的元素个数.如下所示: std::vector<int> vec; ...

  2. copy and swap技巧与移动赋值操作符

    最近在实现一个Delegate类的时候碰到了一个问题,就是copy and swap技巧和移动赋值操作符有冲突. 比如有以下一个类: class Fun { public: Fun(const Fun ...

  3. [CPP] STL 简介

    STL 即标准模板库(Standard Template Library),是 C++ 标准库的一部分,里面包含了一些模板化的通用的数据结构和算法.STL 基于模版的实现,因此能够支持自定义的数据结构 ...

  4. c++ stl swap

    好久没写了,简单水一下吧! 一个是最近没怎么刷题,圣诞,元旦,leetcode暂停的比赛两周,自己最近沉迷于打游戏,没有好好抓住时间. 其实最近看了一点书,是侯捷的 <stl源码剖析>,有 ...

  5. CPP STL学习笔记

    STL的概念 源地址  https://www.ev0l.art/index.php/archives/15/ <li> Iterator (迭代器)<li> Containe ...

  6. 【S17】使用“swap技巧”除去多余的容量

    1.考虑下面的需求,对于vec开始的时候有1000个元素,后来只有10个元素,那么vec的capacity至少还是1000,后面的990个内存单元,没有使用,但是还被vec霸占着.如何释放这些内存呢? ...

  7. [CPP - STL] functor刨根问底儿

    作为STL六大组件之一,在STL源代码及其应用中,很多地方使用了仿函数(functor),尤其在关联型容器(如set.map)以及algorithm(如find_if.count_if等)中.虽然已经 ...

  8. 【转】[STL]vector和deque的内存释放(clear)

    vector的clear成员函数可以清除vector中的元素,使其大小减至0.但它却不能减小vector占用的内存. [cpp] view plain copy int main() { vector ...

  9. Effective STL

    第9条:慎重选择删除元素的方法 删除特定值元素,vector.string.deque用erase-remove:c.erase(remove(c.begin(),c.end(),1963),c.en ...

随机推荐

  1. Codeforces 946 A.Partition

    随便写写,然后写D的题解. A. Partition   time limit per test 1 second memory limit per test 256 megabytes input ...

  2. 网络入侵检测规避工具fragrouter

    网络入侵检测规避工具fragrouter   网络入侵检测系统可以通过拦截数据包,获取内容进而判断是否为恶意数据包.对于传输较大的数据包,通常会采用分片的方式,将大数据包拆分为小数据包进行传输.如果入 ...

  3. CDOJ 879 摩天轮 dp+斜率优化

    原题链接:http://www.acm.uestc.edu.cn/#/problem/show/879 题意: 中文题 题解: 这是一道斜率dp的题. 先把$a$数组排个序. 令$dp[i][j]$表 ...

  4. Java线程池的内部实现

    一.线程池介绍 线程是稀缺资源,如果无限制的创建,不仅会消耗系统资源,还会降低系统的稳定性,合理的使用线程池可以对线程进行统一的分配.调优和监控,并有以下好处: (1)降低资源消耗. (2)提高响应速 ...

  5. Windows7/8/10中无法识别USB设备的问题解决

    1.打开控制面板 [Win+X]->[控制面板] 2.打开设备管理器 首先将面板切换为[小图标] 3.右键卸载“大容量设备”或者“磁盘管理器”的驱动,再重新刷新安装上去

  6. Delphi Helper Record Class

    unit Unit1; {$DEFINE USESGUIDHELP} interface implementation {$IFDEF USESGUIDHELP} uses System.SysUti ...

  7. xcode创建一个工程的多个taget,便于测试和发布多个版本

    背景:很多时候,我们需要在一个工程中创立多个target,也就是说我们希望同一份代码可以创建两个应用,放到模拟器或者真机上,或者是,我们平时有N多人合作开发,当测试的时候,在A这里装了一遍测A写的那块 ...

  8. 邁向IT專家成功之路的三十則鐵律 鐵律十九:IT人待業之道-寬心

    說來很多人可能不相信,筆者從來不把失業當作是一件嚴重的事,相反的我會把它當作是一個很好的轉機.針對一個隨時做好準備的IT人,三個月或半年沒有上班完全沒有甚麼好擔心的.只是如何善用待業的時間,說實在的真 ...

  9. 分布式版本号控制系统Git(二):github

    前言 但凡是喜欢研究技术,或者听大牛们说起过的,都应该至少是听过github这个东西.详细就不介绍了.不了解的能够去了解了解,最基本的功能当然是代码托管啦,上面有各种各样的大牛写的项目. 另外这一章不 ...

  10. php程序调试: xdebug的配置

    怎样在phpeclipse中像调试Java程序一样调试php呢? XDebug的版本号非常多,打开http://xdebug.org/index.php.把站点细致看一下,你会发现有句"If ...