热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

关于STL中vector容器的一些总结

vector作为STL提供的标准容器之一,是经常要使用的,有很重要的地位,并且使用起来也是灰常方便。vector又被称为向量,vector可以形象的描述为长度可以动态改变的数组,功能和数组较为相似

1.vector的简单介绍

vector作为STL提供的标准容器之一,是经常要使用的,有很重要的地位,并且使用起来也是灰常方便。vector又被称为向量,vector可以形象的描述为长度可以动态改变的数组,功能和数组较为相似。实际上更专业的描述为:vector是一个多功能的,能够操作多种数据结构和算法的模板类和函数库,vector之所以被认为是一个容器,是因为它能够像容器一样存放各种类型的对象,简单地说,vector是一个能够存放任意类型的动态数组,能够增加和压缩数据。(注:STL的容器从实现的角度讲可以说是类模板(class teplate)。)

那么vector和数组的主要区别是什么呢??这对于理解vector是很有帮助的~~~~

数组:分配的是静态空间,一般分配了就不可以改变,就像我们熟知的定义了一个数组,那么数组的长度就不可以改变了,我们也不可以进行越界访问,但是编译器不检查越界,这一点在我们编程的时候要尤为注意(很多都可能会烦这样的错误!!)。一般申请的数组长度不能满足我们的要求了,我们要重新申请大一点数组,然后把原数组中数据复制过来。

vector:分配的是动态空间,即:我们发现在声明vector容器的时候也可以不指定容器的大小,vector是随着元素的加入,空间自动扩展的。但是,我们必须要负责任的肯定vector分配的空间是连续的,也就是支持数组中的下标随机访问,实际上vector的实现机制是:预留一部分空间,而且预留空间的大小是按一定比率增长的,如果空间不够用的话,要保证连续,就必须重新new一片空间,然后将原有元素移动到新空间,同时预留新的空间(并且新分配的空间比原来分配的空间),最后将原来的那部分空间释放掉。这样预留空间的好处就是不用每次向vector中加元素都重新分配空间。

2.vecotr容器中常用的函数

2.1.vector容器的构造函数

vector容器的声明方式主要包括一下几种:

--------------------------------------------------------------------------------

vector v   ,创建一个空的vector。

vector v1(v)   ,复制一个vector。

vector v(n)  ,创建一个vector,含有n个数据,数据均已缺省构造产生。

vector v(n, elem)   ,创建一个含有n个elem拷贝的vector。

vector v(beg,end)   ,创建一个以[beg;end)区间的vector。

v.~ vector ()  ,销毁所有数据,释放内存。

--------------------------------------------------------------------------------

下面用一段代码来演示几种常用的声明vector的的方式:

代码如下:

#include
#include

using namespace std;

int main()
{
    vector::iterator iter;
    //第一种方式
    vector v1;
    v1.push_back(1);
    v1.push_back(2);
    v1.push_back(3);
    cout<<"第一种方式的输出结果:"<    for(iter = v1.begin() ; iter != v1.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    //第二种方式
    vector v2(v1);
    cout<<"第二种方式的输出结果:"<    for(iter = v2.begin() ; iter != v2.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    //第三种方式
    vector v3(3);
    cout<<"第三种方式的输出结果:"<    for(iter = v3.begin() ; iter != v3.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    //第四种方式
    vector v4(3,4);
    cout<<"第四种方式的输出结果:"<    for(iter = v4.begin() ; iter != v4.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    //第五种方式
    vector v5(v1.begin(),v1.end()-1);
    cout<<"第五种方式的输出结果:"<    for(iter = v5.begin() ; iter != v5.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    //第六种方式
    int a[] = {1,2,3,4};
    vector v6(a+1,a+2);
    cout<<"第六种方式的输出结果:"<    for(iter = v6.begin() ; iter != v6.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    //
    v6.~vector();
    cout<<"释放内存后的结果是:"<    for(iter = v6.begin() ; iter != v6.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    return 0;
}


运行结果:

小结:注意这种:vector c(beg,end)声明方式,创建一个和[beg;end)区间元素相同的vector,一定要注意是左闭右开区间,同时需要说的是,STL中不论是容器还是算法都是采用的这种左闭右开区间办事的,包括v.end()函数也是返回的vector末端的下位置,相当于int a[n]的a[n],并不能访问~~~

2.2.vector中其他常用的函数用法

--------------------------------------------------------------------------------

v.assign(beg,end)  , 将[beg; end)区间中的数据赋值给v。

v.assign(n,elem)    ,  将n个elem的拷贝赋值给v。

v.at(idx)                ,  传回索引idx所指的数据,如果idx越界,抛出out_of_range。

v.begin()               ,  传回迭代器重的可一个数据。

v.capacity()           ,  返回容器中数据个数。

v.clear()                ,  移除容器中所有数据。

v.empty()              ,  判断容器是否为空。

v.end()                  ,  指向迭代器中的最后一个数据地址。

--------------------------------------------------------------------------------

用上面提到的函数写一个程序演练一下吧:

代码如下:

#include
#include

using namespace std;

int main()
{
    vector::iterator iter;
    vectorv1;
    int a[] = {1,2,3,4};

    //程序段1,练习assign(n,t)
    v1.assign(3,2);
    cout<<"vector 中的元素:";
    for(iter = v1.begin() ; iter != v1.end() ; ++iter)
    {
        cout<<*iter<<" ";
    }
    cout<
    //程序段2,练习assign(beg,end)
    v1.assign(a,a+4);
    cout<<"vector 的长度是:"<    cout<<"vector 中的元素:";
    for(int i = 0 ; i <4 ; ++i)
    {
        cout<    }
    cout<
    //程序段3,练习clear()函数和enpty()函数
    v1.clear();
    if(v1.empty())
    {
        cout<<"vector为空!!!"<    }

    return 0;
}


运行结果:

小结:关于assign函数,对vector变量进行赋值,并且能够自动完成vector大小的修改。

--------------------------------------------------------------------------------

v.insert(pos,elem)         在pos位置插入一个elem拷贝,传回新数据位置(位置指传回地址值)。

v.insert(pos,n,elem)      在pos位置插入在[beg,end)区间的数据。无返回值。

v.insert(pos,beg,end)       在pos位置插入n个elem数据。无返回值。

v.erase(pos)          删除pos位置的数据,传回下一个数据的位置。

v.erase(beg,end)       删除[beg,end)区间的数据,传回下一个数据的位置。

--------------------------------------------------------------------------------

看看vector中的元素的插入和删除操作吧:

代码如下:

#include
#include

using namespace std;

int main()
{
    int a[] = {2,3,4};
    vector v1;
    vector::iterator iter;

    //演示insert函数
    v1.insert(0,1);
    v1.insert(v1.begin()+1,a,a+3);
    v1.insert(v1.begin()+4,2,5);
    cout<<"vector中的数据 :";
    for(iter = v1.begin() ; iter != v1.end() ; ++iter)
    {
        cout<<*iter<<" ";
    }
    cout<    //演示erase函数
    v1.erase(v1.begin(),v1.begin()+2);
    v1.erase(v1.begin()+1);
    cout<<"vector中的数据 :";
    for(iter = v1.begin() ; iter != v1.end() ; ++iter)
    {
        cout<<*iter<<" ";
    }
    cout<    return 0;
}


运行结果:

小结:注意插入和删除操作的pos参数用迭代器传入的。还要注意几种insert函数的返回值。

--------------------------------------------------------------------------------

v.capacity()      返回容器中数据个数。

v.size()        返回容器中实际数据的个数。

v.reserve()     保留适当的容量。

v.resize(num)    重新指定队列的长度。

v.max_size()       返回容器中最大数据的数量。

--------------------------------------------------------------------------------

代码如下:

#include
#include

using namespace std;

int main()
{
    vector v1(4,1);
    vector::iterator iter;
    cout<<"vector的size的值 : "<    cout<<"vector的capacity值 : "<    cout<<"vector的max_size的值 : "<
    //使用reserve函数
    v1.reserve(6);
    cout<    cout<<"vector的size的值 : "<    cout<<"vector的capacity值 : "<    cout<<"vector的max_size的值 : "<    cout<<"vector中的元素是 : ";
    for(iter = v1.begin() ; iter != v1.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<

    //使用resize函数
    v1.resize(6,2);
    cout<    cout<<"vector的size的值 : "<    cout<<"vector的capacity值 : "<    cout<<"vector的max_size的值 : "<    cout<<"vector中的元素是 : ";
    for(iter = v1.begin() ; iter != v1.end() ; iter++)
    {
        cout<<*iter<<" ";
    }
    cout<    return 0;
}


输出结果:

小结:vector 的reserve增加了vector的capacity,但是它的size没有改变!而resize改变了vector的capacity同时也增加了它的size!这是因为:(1)reserve是为容器预留空间,但在空间内不真正创建元素对象,所以在没有添加新的对象之前,不能引用容器内的元素。加入新的元素时,要调用push_back()/insert()函数。(2)resize则是改变容器的大小,且在创建对象,因此,调用这个函数之后,就可以引用容器内的对象了,因此当加入新的元素时,用operator[]操作符,或者用迭代器来引用元素对象。此时再调用push_back()函数,是加在这个新的空间后面的。

--------------------------------------------------------------------------------

c.rbegin()       传回一个逆向队列的第一个数据。

c.rend()          传回一个逆向队列的最后一个数据的下一个位置。

c.pop_back()      删除最后一个数据。

c.push_back(elem)   在尾部加入一个数据。

c.front()          传回地一个数据。

c.back()           传回最后一个数据,不检查这个数据是否存在。

c1.swap(c2)        将c1和c2元素互换。

swap(c1,c2)        同上操作。

--------------------------------------------------------------------------------

这几个函数就比较简单了,这里就不写程序了,有兴趣自己练一下吧!!!


推荐阅读
  • 解决Cydia数据库错误:could not open file /var/lib/dpkg/status 的方法
    本文介绍了解决iOS系统中Cydia数据库错误的方法。通过使用苹果电脑上的Impactor工具和NewTerm软件,以及ifunbox工具和终端命令,可以解决该问题。具体步骤包括下载所需工具、连接手机到电脑、安装NewTerm、下载ifunbox并注册Dropbox账号、下载并解压lib.zip文件、将lib文件夹拖入Books文件夹中,并将lib文件夹拷贝到/var/目录下。以上方法适用于已经越狱且出现Cydia数据库错误的iPhone手机。 ... [详细]
  • 云原生边缘计算之KubeEdge简介及功能特点
    本文介绍了云原生边缘计算中的KubeEdge系统,该系统是一个开源系统,用于将容器化应用程序编排功能扩展到Edge的主机。它基于Kubernetes构建,并为网络应用程序提供基础架构支持。同时,KubeEdge具有离线模式、基于Kubernetes的节点、群集、应用程序和设备管理、资源优化等特点。此外,KubeEdge还支持跨平台工作,在私有、公共和混合云中都可以运行。同时,KubeEdge还提供数据管理和数据分析管道引擎的支持。最后,本文还介绍了KubeEdge系统生成证书的方法。 ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
  • Monkey《大话移动——Android与iOS应用测试指南》的预购信息发布啦!
    Monkey《大话移动——Android与iOS应用测试指南》的预购信息已经发布,可以在京东和当当网进行预购。感谢几位大牛给出的书评,并呼吁大家的支持。明天京东的链接也将发布。 ... [详细]
  • 本文介绍了解决Netty拆包粘包问题的一种方法——使用特殊结束符。在通讯过程中,客户端和服务器协商定义一个特殊的分隔符号,只要没有发送分隔符号,就代表一条数据没有结束。文章还提供了服务端的示例代码。 ... [详细]
  • 基于layUI的图片上传前预览功能的2种实现方式
    本文介绍了基于layUI的图片上传前预览功能的两种实现方式:一种是使用blob+FileReader,另一种是使用layUI自带的参数。通过选择文件后点击文件名,在页面中间弹窗内预览图片。其中,layUI自带的参数实现了图片预览功能。该功能依赖于layUI的上传模块,并使用了blob和FileReader来读取本地文件并获取图像的base64编码。点击文件名时会执行See()函数。摘要长度为169字。 ... [详细]
  • IhaveconfiguredanactionforaremotenotificationwhenitarrivestomyiOsapp.Iwanttwodiff ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 无损压缩算法专题——LZSS算法实现
    本文介绍了基于无损压缩算法专题的LZSS算法实现。通过Python和C两种语言的代码实现了对任意文件的压缩和解压功能。详细介绍了LZSS算法的原理和实现过程,以及代码中的注释。 ... [详细]
  • JVM 学习总结(三)——对象存活判定算法的两种实现
    本文介绍了垃圾收集器在回收堆内存前确定对象存活的两种算法:引用计数算法和可达性分析算法。引用计数算法通过计数器判定对象是否存活,虽然简单高效,但无法解决循环引用的问题;可达性分析算法通过判断对象是否可达来确定存活对象,是主流的Java虚拟机内存管理算法。 ... [详细]
  • 本文介绍了数模国赛的报名参加方法,包括学校报名和自己报名的途径。同时给出了建模竞赛的建议,重在历练的同时掌握方法以及弥补自己的短板。此外,还分享了论文的结构和模型求解部分的注意事项,包括数学命题的表述规范和计算方法的原理等。 ... [详细]
  • 本文介绍了高校天文共享平台的开发过程中的思考和规划。该平台旨在为高校学生提供天象预报、科普知识、观测活动、图片分享等功能。文章分析了项目的技术栈选择、网站前端布局、业务流程、数据库结构等方面,并总结了项目存在的问题,如前后端未分离、代码混乱等。作者表示希望通过记录和规划,能够理清思路,进一步完善该平台。 ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
author-avatar
白变小樱啊
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有