vector以及其使用

news2024/11/18 7:42:01

vector

vector类型是一个标准库中的类型,代表一个容器、集合或者动态数组这样一种概念。既然是容器,那就可以把若干个对象放到里面。当然,这些对象的类型必须相同。简单来说,可以把一堆int型数字放到vector容器中去,复杂点说,可以把一堆相同类型的类对象放到vector容器中去。

所以,如果换个角度考虑,vector能把其他对象装进来,所以称为容器非常合适。容器这个概念经常被提及,读者要知道和理解这个概念。

要想使用这种类型,需要在.cpp源文件开头包含vector头文件:
在这里插入图片描述
另外,为了方便引用这种类型,也要书写:
在这里插入图片描述
定义一个vector类型对象。显然,一旦定义出来,这个对象就是容器了。例如想在里面保存int型数据(容器里面所要装的元素类型),看如下代码:
在这里插入图片描述
上面的代码定义了一个vector类型的对象,名字叫作vjihe,这个对象里面保存的就是int型数据。为什么是int型数据呢?读者可以看到,vector后面有一对“<>”,“<>”里面是int,表示这个vector类型的对象(容器)里面存放的是int型对象(int型数据/元素)。

这个<int>的写法读者可能第一次见到,会觉得是一种奇怪的写法,在后面章节中会讲到“类模板”的概念,其实vector就是一个类模板,这里的“<>”实际上是类模板的一个实例化过程。但是类模板的实例化过程眼下对于读者来讲,理解起来还比较生涩,后面学习模板的时候再详细阐述,所以这里笔者换一种说法来帮助读者理解类模板。

vector理解成一个残缺的类类型,这意味着使用时光有类名vector还不够,还需要额外给vector类模板提供其要在其中保存什么类型数据的信息,这个信息就是通过<int>来提供(模板名后跟一对“<>”,“<>”内放入类型信息),所以,在使用vector时,一定要在它后面跟一对“<>”并在其中跟一个该vector容器中要保存的数据(元素)类型的信息,这才算一个完整的类型(完整的类类型)。例如,vector不是一个完整类型,而vector<int>却是一个完整的类型。看看如下范例:
在这里插入图片描述
一般来讲,vector容器里面可以装很多种不同类型的数据作为其元素(容器中装的内容简称“元素”)。看看如下范例:
在这里插入图片描述
但是vector不能用来装引用。请记住,引用只是一个别名,不是一个对象。所以,下面的写法会报语法错误:
在这里插入图片描述

定义和初始化vector对象

(1)空vector。
定义如下:

vector<string> mystr;
//创建一个string类型的空vector对象(容器),
//现在mystr里不包含任何元素

后续就可以用相关的一些操作函数往这个空对象里增加数据了。

例如,可以往这个容器的末尾增加一些数据。这里可以使用vector的成员函数push_back往容器末尾增加数据。看看如下范例,注意看它的下标[0],[1],[2],…不断增长,如图13.3所示。

mystr.push_back("abcd");
mystr.push_back("def");

(2)在vector对象元素类型相同的情况下,进行vector对象元素复制(新副本)。

vector<string> mystr2(mystr);	//把mystr元素复制给了mystr2
vector<string>mystr3 = mystr;	//把mystr元素复制给了mystr3

(3)在C++11中,还可以用初始化列表方法给初值,这个时候用“{}”括起来。

vector<string> def = {"aaa","bbb","ccc"};

当然“{}”里面为空也可以,那就相当于没有初始化,是一个空的vector了。

(4)创建指定数量的元素。请注意,有元素数量概念的初始化,用的都是“()”。

如果不给元素初值,那么元素的初值要根据元素类型而定,例如元素类型为int,系统给的初值就是0,元素类型为string,系统给的初值就是"",但也存在有些类型,必须给初值,否则就会报错。
如下范例演示不给元素初值的情况:

vector<int> ijihe2(20);	//20个元素,下标[0]~[19],每一个元素值都为0
vector<string> sjihe2(5);	//15个元素,下标[0]~[4],每一个元素值都为""

(5)多种初始化。“()”一般表示对象中元素数量这种概念,“{}”一般表示元素的内容这种概念,但又不是绝对。看看如下范例:
在这里插入图片描述

vector对象上的操作

其实,在使用vector时,最常见的情况是并不知道vector里会有多少个元素,使用时会根据需要动态地增加和减少。所以一般来讲,使用者是先创建一个空的vector对象,然后通过代码向这个vector里增加或减少元素。这里将要介绍一些vector类型提供的常用方法。vector上很多的操作和string很相似。
(1)判断是否为空empty(),返回布尔值。

   vector<int> ivec;
    if (ivec.empty())   //条件成立
    {   
        cout<<"ivec为空"<<endl;
    }

(2)push_back:一个非常常用的方法,用于向vector末尾增加一个元素。

    vector<int> ivec;  //先声明成空的vector对象
    ivec.push_back(1);
    ivec.push_back(2);
    for (int i=3;i<=100;i++)
    {
    ivec.push_back(i);
    }

在上面的范例中,注意观察,能够发现,值2在值1的后面(最后插入的元素在vector容器的最末尾)。调试结果如下图所示。

(3)size:返回元素个数。
(4)clear:移除所有元素,将容器清空。
(5)v[n]:返回v中的第n个字符(n是一个整型值),位置从0开始计算,位置值n也必须小于.size(),如果下标引用超过这个范围,或者用下标访问一个空的vector,都会产生不可预测的结果(因为编译器可能发现不了这种错误)
在这里插入图片描述
(6)赋值运算符(=)。

    vector<int> ivec;
    //先声明成空的vector对象
    ivec.push_back(1);
    ivec.push_back(2);
    for (int i=3;i<=100;i++)
    {
        ivec.push_back(i);
    }
    vector<int> ivec2;
    ivec2.push_back(111);
    ivec2=ivec;//也得到了100个元素,用ivec中的内容取代了ivec2中原有内容,上行这个111就
               //被冲掉了
    ivec2={12,13,14,15};
    //用{}中的值取代了ivec2原有值
    cout<<ivec2.size()<<endl;  //4

(7)相等和不等(==和!=)。
两个vector对象相等:元素数量相同,对应位置的元素值也都相同。否则就是不相等。
在这里插入图片描述
(8)范围for的应用:和讲解string时对范围for的应用类似

    vector<int> vecvalue{1,2,3,4,5 };
    for(auto& vecitem:vecvalue)  //为了修改vecvalue内部值,这里是引用,引用会绑定到元素上,达
                                //到通过引用改变元素值的目的
    
        vecitem *= 2;       //扩大一倍
        
    for (auto vecitem : vecvalue)
    cout << vecitem << endl;

针对范围for语句,这里希望引申一步进行讲解。如果在范围for中,增加改变vector容量的代码,则输出就会变得混乱:

    vector<int> vecvalue{1,2,3,4,5 };
    for (auto vecitem : vecvalue)
    {
        vecvalue.push_back(888);
        cout << vecitem << endl;

    }

范围for,在这里用来遍历vector容器中的元素。这里的vecitem是定义的一个变量,后面的vecvalue是一个序列(容器),for语句中使用auto来确保序列中的每个元素都能够转换成变量vecitem对应的类型,所以一般在范围for语句中习惯使用auto(编译器来指定合适的vecitem类型)。

那为什么上述代码会产生混乱的输出呢?

因为每次执行for循环,都会重新定义vecitem,并且把它的值初始化成vecvalue序列中的下一个值。在刚刚进入这个for循环时,在系统内部会记录序列结束的位置值,但一旦在这个范围for里面改动这个序列的容量(如增加/删除元素),那么这个序列结束的位置值就肯定会发生改变,这个改变会导致for语句的混乱,其输出的值也就乱了。

所以,请记住一个结论,在for语句中,不要改变vector的容量,增加、删除元素都不可以。请读者千万千万不要写出这种错误代码,否则隐患无穷,切记切记!

迭代器精彩演绎、失效分析及弥补、实战

迭代器简介

迭代器是一个经常听到和用到的概念。上一节学习了vector,笔者说过,这是一个容器,里面可以容纳很多对象。那迭代器是什么呢?迭代器是一种遍历容器内元素的数据类型。这种数据类型感觉有点像指针,读者就理解为迭代器是用来指向容器中的某个元素的。

string可以通过“[]”(下标)访问string字符串中的字符,vector可以通过“[]”访问vector中的元素。但实际上,在C++中,很少通过下标来访问它们,一般都是采用迭代器来访问。

除了vector容器外,C++标准库中还有几个其他种类的容器。这些容器都可以使用迭代器来遍历其中的元素内容。string其实是字符串,不属于容器,但string也支持用迭代器遍历。

通过迭代器,可以读取容器中的元素值、修改容器中某个迭代器所代表(所指向)的元素值。此外,迭代器可以像指针一样——通过++、–等运算符从容器中的一个元素移动到另一个元素。

许多容器如上述的vector,在C++标准库中,还有其他容器如list、map等都属于比较常用的容器,C++标准库为每个这些容器都定义了对应的一种迭代器类型,有很多容器不支持“[]”操作,但容器都支持迭代器操作。写C++程序时,笔者也强烈建议读者不要用下标访问容器中的元素,而是用迭代器来访问容器中的元素。

容器的迭代器类型

刚刚讲过,C++标准库为每种容器都定义了对应的迭代器类型。这里就以容器vector为例,演示一下:

    vector<int> iv = {100,200,300};     //定义一个容器
    vector<int >:: iterator iter;       //定义迭代器,也必须是以vector<int>开头

上面的语句是什么意思呢?后面这条语句定义了一个名为iter的变量(迭代器),这个变量的类型是vector<int>::iterator类型,请注意这种写法“::iterator”。iterator是什么?它是每个容器(如vector)里面都定义了的一个成员(类型名),这个名字是固定的,请牢记。
在理解的时候,就把整个vector<int>::iterator理解成一种类型,这种类型就专门应用于迭代器,当用这个类型定义一个变量的时候,这个变量就是一个迭代器。

迭代器begin/end、反向迭代器rbegin/rend操作

1.迭代器

每一种容器,如vector,都定义了一个叫begin的成员函数和一个叫end的成员函数。这两个成员函数正好用来返回迭代器类型。看看如下范例。
(1)begin返回一个迭代器类型(就理解成返回一个迭代器)。
在这里插入图片描述
(2)end返回一个迭代器类型(就理解成返回一个迭代器)。
在这里插入图片描述
对上面的代码进行跟踪调试,观察begin和end结果可以看到,end()指向了一个乱数字,如图13.5所示。

    vector<int> iv = {100,200,300};     //定义一个容器
    vector<int >::iterator iter;       //定义迭代器,也必须是以vector<int>开头
    iter = iv.begin();
    iter = iv.end();

在这里插入图片描述
(3)如果容器为空,则begin返回的迭代器和end返回的迭代器相同。看看如下范例:

    vector<int> iv2;
    vector<int>::iterator iterbegin = iv2.begin();
    vector<int>::iterator iterend = iv2.end();
    if (iterbegin == iterend)   //条件成立
    {
        cout << "容器为空" << endl;
    }

在这里插入图片描述

所以,end返回的迭代器并不指向容器vector中的任何元素,它起到实际上是一个标志(岗哨)作用,如果迭代器从容器的begin位置开始不断往后游走,也就是不断遍历容器中的元素,那么如果有一个时刻,iter走到了end位置,那就表示已经遍历完了容器中的所有元素。
(4)写一段代码,传统的通过迭代器访问容器中元素的方法如下:

    vector<int>iv={100,200,300};    //定义一个容器
    //经典传统用法,这里用++、!=等运算符来对迭代器进行操作
    for (vector<int>::iterator iter = iv.begin();iter != iv.end();iter++)
    {
        cout << *iter << endl;
    }

运行起来看结果:100、200、300。

2.反向迭代器

如果想从后面往前遍历一个容器,那么,用反向迭代器就比较方便。反向迭代器使用的是rbegin成员函数和rend成员函数。
(1)rbegin返回一个反向迭代器类型,指向容器的最后一个元素。
(2)rend返回一个反向迭代器类型,指向容器的第一个元素的前面位置。
rbegin和rend成员函数指向的容器位置示意图如图13.7所示。
在这里插入图片描述
看看如下范例:

    vector<int>iv={100,200,300};
    for (vector<int>::reverse_iterator riter = iv.rbegin(); riter != iv.rend(); riter++)
    {
        cout<<*riter<<endl;
    }
    

运行起来看结果:300、200、100。

迭代器运算符

(1)iter:返回迭代器iter所指向元素的引用。必须要保证该迭代器指向的是有效的容器元素,不能指向end,因为end是末端元素后面的位置,也就是说,end已经指向了一个不存在元素。前面的cout≪iter≪endl;就是使用*iter的演示范例,这里不做进一步演示了。
(2)++iter:和iter++是同样的功能——让迭代器指向容器中的下一个元素。但是已经指向end的迭代器,不能再++,否则运行时报错。
(3)–iter:和iter–是同样的功能——让迭代器指向容器中的前一个元素。了解++自然也就能了解–。看看如下范例:
在这里插入图片描述
(4)iter1==iter2或iter1!=iter2:判断两个迭代器是否相等。如果两个迭代器指向的是同一个元素,就相等,否则就不相等。
(5)结构成员的引用。看看如下范例:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/984608.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Ubuntu之apt-get系列--安装JDK8--方法/教程

原文网址&#xff1a;Ubuntu之apt-get系列--安装JDK8--方法/教程_IT利刃出鞘的博客 简介 本文介绍如何在Ubuntu下安装JDK8。 验证是否安装 可以通过如下命令判断系统是否有安装ssh服务&#xff1a; 命令 java -version 结果 如上所示&#xff0c;表示还没有安装。 查看…

自动化运维工具Ansible教程(一)【入门篇】

文章目录 前言Ansible 入门到精通入门篇进阶篇精通篇入门篇1. Ansible 简介2. 安装 Ansible1. 通过包管理器安装&#xff1a;2. 通过源码安装&#xff1a; 3. Ansible 的基本概念和核心组件4. 编写和运行第一个 Ansible Playbook5. 主机清单和组织结构主机清单组织结构 6. Ansi…

好用免费的Chat GPT(亲测有用)

1、MindLink麦灵 MindLink麦灵 点进登录后 普通用户可以提问100次 2、你问我答 你问我答 无限次数的。 3、灵感 灵感 点击链接后会提示你如何下载使用。 这个有win版和mac版&#xff0c;点击登陆后&#xff0c;每日都会有30次GPT3/3.5的提问。 4、WebTab 在浏览器插件中…

如何加载卫星影像全国一张图?

我们在“你的家乡清晰可见&#xff0c;全国卫星影像100%覆盖&#xff01;”一文中&#xff0c;为你分享了一个可以十分方便地查看家乡高清卫星影像的方法。 该卫星影像数据源由长光卫星提供&#xff0c;你可以通过水经微图PC端或Web端查看&#xff0c;也可以将该卫星影像加载到…

指针跃动——客户运营服务中心上线了!

指针跃动——客户运营服务中心上线了&#xff01; ——打通客户运营服务全链路—— 随着全国代驾业务需求的不断增长&#xff0c;“指针跃动”宣布&#xff1a;指针跃动——客户运营服务中心上线了&#xff01; 以新的思维方式来看待客户服务&#xff0c;利用人工智能、大数据等…

Flyway-数据库管理工具使用与命令

工作流程 应用程序完成数据库连接池的建立后&#xff0c;flyway自动运行初次使用&#xff0c;flyway会创建一个flyway_schema_history表&#xff0c;用于记录sql执行记录flyway会扫描项目指定路径&#xff08;默认classpath: db/migration&#xff09;下的所有sql脚本&#xf…

Magisk V26.3卡刷包APK最新版下载-支持payload.bin自动维补ROOT

从magisk V26.0开始&#xff0c;topjohnwu作者更新的比较快&#xff0c;从26.0.-26.3各种版本快速迭代 今天又更新到了最新的26.3版本。从界面来看&#xff0c;和之前没什么不同&#xff0c;最大的可能就是功能 上的新增。从官方日志中&#xff0c;我们可以清晰的看到26.3版本三…

ES6之 变量的解构赋值 ➕ 扩展运算符(…)

ES6之 变量的解构赋值 ➕ 扩展运算符 1. 变量的解构赋值2. 扩展运算符2.1 简介&#xff08;官网&#xff09;2.2 应用例子2.2.1 简单例子12.2.2 数组拷贝2.2.3 连接多个数组2.2.4 拷贝对象&#xff08;多层和多维数组一样&#xff09;2.2.5 合并对象2.2.6 关于展开对象&#xf…

Python爬虫数据存哪里|数据存储到文件的几种方式

前言 大家早好、午好、晚好吖 ❤ ~欢迎光临本文章 爬虫请求解析后的数据&#xff0c;需要保存下来&#xff0c;才能进行下一步的处理&#xff0c;一般保存数据的方式有如下几种&#xff1a; 文件&#xff1a;txt、csv、excel、json等&#xff0c;保存数据量小。 关系型数据库…

【代码分享不会用?玩转Git,跟上节奏 快上车!】

一、安装 Git 客户端 这里为大家提供了windows版的Git客户端以及安装图文详解文档。百度网盘&#xff1a; https://pan.baidu.com/s/1CDu0Ke199pt3Ysv-QtWObA 提取码&#xff1a;8888 如果过期了请留言联系我。 二、注册码云账号 打开码云网站&#xff1a;https://gitee.com…

在Git中将本地分支推送到远程仓库

这里很明显 我git云端只有一个master分支 然后 我在本地创建了一个develop分支 然后 现在我想将他放在云端 首先 我们要执行 git checkout -b develop将本地切换到 develop 分支上 因为我这里已经选择的就是了 就不需要了 然后我们执行 git push origin develop这样 刷新云…

基于Java+SpringBoot+Vue前后端分离小学生身体素质测评管理系统设计和实现

博主介绍&#xff1a;✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专…

高压放大器在机械制造领域的应用有哪些

在机械制造领域&#xff0c;高压放大器扮演着至关重要的角色。它们被广泛应用于各种机械设备和系统中&#xff0c;提供高压力、高精度的电信号放大。下面安泰电子将详细介绍高压放大器在机械制造领域的几个关键应用。 材料测试和强度试验 高压放大器广泛应用于材料测试和强度试…

手游联运平台都具备哪些功能?

手游联运平台是为了方便游戏发行商进行游戏发行和运营而提供的一种服务平台&#xff0c;具备多种功能以支持游戏在不同渠道上的推广和运营。以下是一些手游联运平台通常具备的功能&#xff1a; 多渠道发行&#xff1a;提供多种渠道&#xff0c;如应用商店、社交媒体、合作伙伴等…

回归与聚类算法系列②:线性回归

目录 1、定义与公式 2、应用场景 3、特征与目标的关系分析 线性回归的损失函数 为什么需要损失函数 损失函数 ⭐如何减少损失 4、优化算法 正规方程 梯度下降 优化动态图 偏导 正规方程和梯度下降比较 5、优化方法GD、SGD、SAG 6、⭐线性回归API 7、实例&#…

Deep Java Library(五)DJLServing java client demo

1.工具类HttpUtils package com.lihao.client; import java.io.IOException; import java.net.URI; import java.net.URLEncoder; import java.net.http.HttpClient; import java.net.http.HttpRequest; import java.net.http.HttpResponse; import java.nio.charset.Standard…

2023 年高教社杯全国大学生数学建模竞赛题目 A 题 定日镜场的优化设计

A 题 定日镜场的优化设计 构建以新能源为主体的新型电力系统&#xff0c;是我国实现“碳达峰”“碳中和”目标的一项重要措施。塔式太阳能光热发电是一种低碳环保的新型清洁能源技术[1]。 定日镜是塔式太阳能光热发电站&#xff08;以下简称塔式电站&#xff09;收集太阳能的基…

WorkPlus Meet白板和文档共享功能上线,私有化视频会议全新升级

在迅猛发展的数字化时代&#xff0c;私有化视频会议成为企业高效沟通和协作的关键工具。WorkPlus Meet作为领先品牌&#xff0c;倾力打造私有化视频会议平台&#xff0c;并且最新上线了全新的白板和文档共享模块。本文将重点介绍WorkPlus Meet如何通过创新功能和稳定性&#xf…

金蝶迷路“云”丛中

​ 来源 | 科技新知 作者 | 樟稻 在云转型的浪潮中&#xff0c;金蝶曾一度领先用友数年&#xff0c;如今却在大模型上出现被逆转的迹象。 8月&#xff0c;金蝶推出自主研发的金蝶云苍穹GPT大模型&#xff0c;号称填补了国内第一个财务大模型的空白。而此前&#xff0c;用友…

【Axure教程】中继器网格拖动摆放

拖动摆放图标在移动端操作中扮演了重要的角色&#xff0c;允许用户自定义其设备的界面&#xff0c;使其更符合其偏好和使用习惯。这可以提高用户对设备的满意度和舒适度&#xff0c;将最常用的应用放置在易于访问的位置&#xff0c;从而提高使用效率。所以拖动摆放这类型操作不…