一、容器vector

使用vector你必须包含头文件<vector>:

#include<vector>

型别vector是一个定义于namespace std内的template:

  1. template<class _Ty,
  2. class _Ax = allocator<_Ty> >

第二个參数定义内存模型。

我们一般採用默认的内存模型。

二、vector的功能

vector模塑出一个动态数组。vector将其元拷贝到内部的动态数组中。

元素之间总是存在某种顺序,它是一种有序群集。

支持随即存取。

它的迭代器是随机存取迭代器。所以对不论什么一个STL算法都奏效。

向vector加入一个元素或者删除当中的一个元素,其后的全部元素都要移动位置,每一次移动都要调用赋值操作符。

所以。在末端加入或删除元素,性能非常好。

可是在前段或者中部的话。性能较差。

vector优异性能的秘诀之中的一个是它配置比其所容纳的元素所需很多其它的内存。我们须要了解大小容量的关系。

函数size()能够返回vector的大小。即vector中实际元素的个数。

而capacity()返回容量。是当前的vector所实际可以容纳的元素的数量。它应该总是大于或者等于vector的大小。

假设须要向vector中放置比capacity很多其它的元素。则须要又一次配置内部存储器。

vector的容量也会随之增长。

看以下的演示样例代码:

  1. #include <iostream>
  2. #include <vector>
  3. #include <string>
  4. #include <algorithm>
  5. using namespace std;
  6. int main()
  7. {
  8. vector<string> sentence(1);
  9. cout << "max_size():" << sentence.max_size() << endl;
  10. cout << "size():" << sentence.size() << endl;
  11. cout << "capacity():" << sentence.capacity() << endl;
  12. sentence.reserve(5);
  13. sentence.push_back("Hello,");
  14. sentence.push_back("how ");
  15. sentence.push_back("are ");
  16. sentence.push_back("you ");
  17. sentence.push_back("?");
  18. copy(sentence.begin(),sentence.end(),
  19. ostream_iterator<string>(cout," "));
  20. cout << endl;
  21. cout << "max_size():" << sentence.max_size() << endl;
  22. cout << "size():" << sentence.size() << endl;
  23. cout << "capacity():" << sentence.capacity() << endl;
  24. swap(sentence[1],sentence[3]);
  25. sentence.insert(find(sentence.begin(),sentence.end(),"?"),
  26. "always");
  27. sentence.back() = "!";
  28. copy(sentence.begin(),sentence.end(),
  29. ostream_iterator<string>(cout," "));
  30. cout << endl;
  31. cout << "max_size():" << sentence.max_size() << endl;
  32. cout << "size():" << sentence.size() << endl;
  33. cout << "capacity():" << sentence.capacity() << endl;
  34. }

执行结果:



在程序中。当再次向vector插入元素时。因为vector的容量不够,所以引起了内存的又一次分配。可是capacity()的结果与实作版本号有关,max_size也是。

vector的容量十分重要,是由于:

1、一旦内存又一次配置,与之相关的全部的reference、pointers、iterators都会失效。

2、内存配置非常费时。

解决问题的方法有:

1、能够使用reserve()保留适当容量,降低又一次配置内存的次数。演示样例代码:

  1. vector<string> sentence(1);
  2. sentence.reserve(50);

2、在初始化期间向构造函数传递附加參数。构造出足够的空间。

  1. vector<T> v(5);

当然,这样的元素的型别必须提供默认构造函数。可是假设元素的型别比較复杂,初始化操作也非常耗时。

假设仅仅是为了保留足够的内存。用法1较好。

注意:reserve不能缩减vector的容量。由此,我们能够知道。即使删除元素。其reference、pointers、iterators也会继续有效,指向动作发生前的位置。

可是插入操作可能使reference、pointers、iterators失效(由于可能会导致又一次配置空间)。

使用swap函数能够缩减vector容量。

由于两个vector交换内容后,他们的容量也会互换。

1、

  1. template<class T>
  2. void shrinkCapacity(vector<T> &v)
  3. {
  4. vector<T> tmp(v);
  5. v.swap(tmp);
  6. }

2、

  1. vector<T>(v).swap(v);

上面两种方法等价。

都是先构造出一个暂时vector对象,以v的元素进行初始化,再与v进行交换。

须要注意的是:暂时对象一般都是精确分配实际所需的内存。

所以可以起到减小vector容量的效果。

三、vector的操作函数

全部的构造函数和析构函数例如以下:

非变动性操作:

赋值操作:

上述操作进行的是将新元素赋值给vector,并将旧元素所有移除!演示样例代码:

  1. #include <iostream>
  2. #include <vector>
  3. #include <string>
  4. #include <algorithm>
  5. using namespace std;
  6. int main()
  7. {
  8. vector<string> sentence(1);
  9. cout << "max_size():" << sentence.max_size() << endl;
  10. cout << "size():" << sentence.size() << endl;
  11. cout << "capacity():" << sentence.capacity() << endl;
  12. sentence.reserve(5);
  13. sentence.push_back("Hello,");
  14. sentence.push_back("how ");
  15. sentence.push_back("are ");
  16. sentence.push_back("you ");
  17. sentence.push_back("?

    ");

  18. copy(sentence.begin(),sentence.end(),
  19. ostream_iterator<string>(cout," "));
  20. cout << endl;
  21. sentence.assign(3,"new");
  22. copy(sentence.begin(),sentence.end(),
  23. ostream_iterator<string>(cout," "));
  24. cout << endl;
  25. }

执行结果:

能够看出原来的元素所有被删除了。

元素存取

在这几个函数中,唯一进行下标检查的是at函数。

因此。在调用operator[]的时候,必须心理清楚索引是否是有效的。

迭代器相关函数

迭代器失效的两种情况是:

1、在一个较小的位置上删除或者是移动元素。

2、因为容量的变换引起内存又一次分配。

插入和移除元素

插入和移除元素。都会使“作用点”之后的各元素的reference、pointers、iterators失效。插入操作还可能引发内存又一次分配,那么该容器上的全部的reference、pointers、iterators都会失效。

四、把vector当做一般数组使用

如今的C++标准保证vector的元素必须分布于连续空间中。对于vector中的一个合法索引,满足下列表达式:

&v[i] = &v[0] + i;

我们必须保证vector可以容纳全部数据。

假设使用的是C-String,记住最后有个'\0'。

仅仅要我们须要一个元素型别为T的数组,就能够採用vector<T>,然后传递第一个元素的地址给它。

注意:千万不要把迭代器当做第一元素的地址来传递。由于vector迭代器是由实作版本号定义的,不一定是一个一般指针。

  1. printf("%s",v.begin());//ERROR(might work,but not portable)
  2. printf("%s",&v[0]);//OK

STL源代码学习--vector用法汇总的更多相关文章

  1. STL源代码学习(vector篇)

    #include <concept_checks.h> #include<stl_allocate.h> /// The vector base class's constru ...

  2. C++ STL源代码学习(map,set内部heap篇)

    stl_heap.h ///STL中使用的是大顶堆 /// Heap-manipulation functions: push_heap, pop_heap, make_heap, sort_heap ...

  3. C++ STL源代码学习之算法篇

    ///因为篇幅太长,因此,删去了非常多接口,仅仅分析了内部实现,算法对迭代器的要求也被删去 /// search. template <class _ForwardIter1, class _F ...

  4. C++ STL常用容器基本用法汇总

    1.vector 包含头文件#include<vector> 使用命名域using namespace std 定义元素类型为T的vector vector<T> vec 增: ...

  5. C++ STL源代码学习(list篇)

    ///STL list为双向循环链表 struct _List_node_base { _List_node_base* _M_next; _List_node_base* _M_prev; }; t ...

  6. C++ STL 源代码学习(之deque篇)

    stl_deque.h /** Class invariants: * For any nonsingular iterator i: * i.node is the address of an el ...

  7. C++STL源代码学习(之slist篇)

    ///stl_slist.h ///list为双向循环链表,slist为单向链表.某些操作效率更高 ///slist是SGI额外提供的单向链表,不属于C++标准 struct _Slist_node_ ...

  8. STL vector用法介绍

    STL vector用法介绍 介绍 这篇文章的目的是为了介绍std::vector,如何恰当地使用它们的成员函数等操作.本文中还讨论了条件函数和函数指针在迭代算法中使用,如在remove_if()和f ...

  9. STL中的Vector相关用法

    STL中的Vector相关用法 标准库vector类型使用需要的头文件:#include <vector>. vector 是一个类模板,不是一种数据类型,vector<int> ...

随机推荐

  1. Iptables-主机防火墙设置

    基于Iptables构建主机防火墙 Iptables优点: 数据包过滤机制,它会对数据包包头数据进行分析. 1.1.1 加载相关薄块到内核 [root@centos7 ~]# lsmod | egre ...

  2. php网站修改为https后curl报错301

    今日测试项目时需调用post模拟传参测试接口是否可用,但返回报错信息(301永久迁移),在网上搜寻解决办法无果,最后发现只要将跳转地址修改为https://+url的形式就可以了

  3. 【例 7-12 UVA - 1343】The Rotation Game

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 迭代加深搜索. 每次抽动操作最多只会让中间那一块的区域离目标的"距离"减少1. 以这个作为剪枝. 枚举最大深度. ...

  4. 项目太大tomcat启动不起来

    双击server,Open launch configuration Arguments VM arguments增加参数: -Xms512m -Xmx1024m -Xss4m -XX:PermSiz ...

  5. 洛谷 P2069 松鼠吃果子

    P2069 松鼠吃果子 题目描述 有N个一种松鼠喜欢吃的果子由下向上串排成一列,并标号1,2,...N.一只松鼠从最下果子开始向上跳,并且第i次跳可以一次跳过i*i*i除以5的余数+1个果子(=i*i ...

  6. HTTP详解--请求、响应、缓存

    1. HTTP请求格式 做过Socket编程的人都知道,当我们设计一个通信协议时,“消息头/消息体”的分割方式是很常用的,消息头告诉对方这个消息是干什么的,消息体告诉对方怎么干.HTTP协议传输的消息 ...

  7. GO语言学习(十一)Go 语言循环语句

    Go 语言提供了以下几种类型循环处理语句: 循环类型 描述 for 循环 重复执行语句块 循环嵌套 在 for 循环中嵌套一个或多个 for 循环 语法 Go语言的For循环有3中形式,只有其中的一种 ...

  8. (转)yum & wget代理设置

    转自 http://www.cnblogs.com/windows/archive/2012/12/14/2817533.html   yum  配置代理服务器访问要设置所有 yum 操作都使用代理服 ...

  9. sbt教程

      更详细内容请见:http://www.scala-sbt.org/0.13/tutorial/Basic-Def.html 或者 http://wenku.baidu.com/link?url=o ...

  10. SpringMvc(4-1)Spring MVC 中的 forward 和 redirect(转)

    Spring MVC 中,我们在返回逻辑视图时,框架会通过 viewResolver 来解析得到具体的 View,然后向浏览器渲染.通过配置,我们配置某个 ViewResolver 如下: <b ...