【QT】容器类的迭代

迭代器(iterator)为访问容器类里的数据项提供了统一的方法,Qt有两种迭代器类:Java类型的迭代器和STL类型的迭代器。
Java类型的迭代器更易于使用,且提供一些高级功能,而STL类型的迭代器效率更高。
Qt还提供一个关键字foreach(实际是<QtGlobal>里定义的一个宏)用于方便地访问容器里所有数据项。

1.JAVA类型迭代器

1.1 JAVA类型迭代器总表

对于每个容器类,有两个Java类型迭代器:一个用于只读操作,一个用于读写操作。
Map和QHash等关联容器类的迭代器用法相同,QList和QLinkedList、QSet等容器类的用
法相同,所以下面只以QMap和QList为例介绍迭代器的用法。Java类型的迭代器如表3-3所示。
Java类型迭代器的指针不是指向一个数据项,而是在数据项之间,迭代器指针位置示意图如图所示。
图1-1  JAVA类型迭代器位置示意图

1.2 顺序容器类的迭代器的使用

1.2.1 Java类型QList容器类迭代器

(1)QListIterator常用函数
QListIterator用于移动指针和读取数据的函数见表3-4。
(2)正向遍历QList<QString>容器所有数据
QList<QString>容器对象list作为参数传递给QListlterator<QString>迭代器i的构造函数,i
用于对list作只读遍历。起始时刻,迭代器指针在容器第一个数据项的前面(图1-1中数据项"A”
的前面),调用hasNext()判断在迭代器指针后面是否还有数据项,如果有,就调用next()跳过一个
数据项,并且next()函数返回跳过去的那个数据项。
QList<QString>list;
list<<"A"<<"B"<<"C"<<"D";
QListIterator<QString>i(list);
while(i.hasNext())
    qDebug()<<i.next();
(2)反向遍历QList<QString>容器所有数据
QListIterator<QString>i(list);
i.toBack();
while(i.hasprevious())
    qDebug()<<i.previous;
代码和正序遍历是对称的,我们调用toBack()将迭代器移到最后一项后面的位置。下图描述了在一
个迭代器上调用next()和previous()函数的效果:
(3)QMutableListIterator遍历修改容器数据
QList<int>list;
list<<1<<2<<3<<4<<5;
QMutableListIterator<int>i(list);
while(i.hasNext()){
    if(i.next()%2 != 0)
        i.remove();//remove()函数移除next()函数刚刚跳过的一个数据项,不会使迭代器失效。
    if(i.next() > 128)
        i.setValue(128);//setValue()函数可以修改刚刚跳过去的数据项的值
}

1.3 关联容器类的迭代器的使用

1.3.1 Java类型QMap容器类迭代器

对于关联容器类QMap<KeyT>,使用QMapIterator和QMutableMapIterator迭代器类,它们具
有表3-4所示的所有函数,主要是增加了 key()和value()函数用于获取刚刚跳过的数据项的键和值。
例如,下面的代码将删除键(城市名称)里以“City”结尾的数据项。
QMap<QString,QString>map;
map.insert("Paris","France");
map.insert("New York","USA");
maP.insert("Mexico City","USA");
map·insert("Moscow","Russia");
......
QMutab1eMapIterator<QString,QString>i(map);
while(i.hasNext()){
    if(i.ne×().key().endsWith("City"))
        i.remove();
}
如果是在多值容器里遍历,可以用 findNext()或findPrevious()查找下一个或上一个值,如下面
的代码将删除上一示例代码中map里值为"USA”的所有数据项。
QMutableMapIterator<QString,QString>i(map);
while(i.findNext("USA"))
    i.remove();

2.STL类型迭代器

2.1 STL类型迭代器总表

STL迭代器与Qt和STL的原生算法兼容,并且进行了速度优化。具体类型见表3-5。
对于每一个容器类,都有两个STL类型迭代器:一个用于只读访问,一个用于读写访问。无需修改数据时一定使用只读迭代器,因为它们速度更快。
注意:在定义只读迭代器和读写迭代器时的区别,它们使用了不同的关键字,const_iterator定义只读迭代器,iterator定义读写迭代器。此外,还可以使用const_reverse_iterator和reverse_iterator定义相应的反向迭代器。
STL类型的迭代器是数组的指针,所以“++”运算符使迭代器指向下一个数据项,“*”运算符返回数据项内容。与Java类型的迭代器不同,STL迭代器直接指向数据项,STL迭代器指向位置示意图如图2-1所示。
图2-1 STL类型迭代器位置示意图
begin()函数使迭代器指向容器的第一个数据项,end()函数使迭代器指向一个虚拟的表示结尾的数据项,end()表示的数据项是无效的,一般用作循环结束条件。下面仍然以QList和QMap为例说明sTL迭代器的用法,其他容器类迭代器的用法类似。

2.2 顺序容器类的迭代器的使用

2.2.1 STL类型QList容器类迭代器

(1)正向只读迭代器
QList<QString>list;
list<<"A"<<"B"<<"C"<<"D";
QList<QString>::const_iterator i;
for(i = list.constBegin();i != list.constEnd();++i)
    qDebug()<<*i;
(2)反向读写迭代器
constBegin()和constEnd()是用于只读迭代器的,表示起始和结束位置。
若使用反向读写迭代器,并将上面示例代码中list的数据项都改为小写,代码如下:
QLIstanbul<QString>::reverse_iterator i;
for(i =list.rbegin();i != list.rend();++i)
    *i = i->toLower();//修改数据

2.3 关联容器类的迭代器的使用

2.3.1 STL类型QMap容器类迭代器

对于关联容器类QMap和QHash,迭代器的“*”操作符返回数据项的值。如果想返回键,使用key()函数。对应的,用value()函数返回一个项的值。
例如,下面的代码将QMap<int,int>map中所有项的键和值输出。
QMap<int,int>map;
......
QMap<int,int>::const_iterator;
for(i = list.constBegin();i != list.constEnd();++i)
    qDebug()<<i.key()<<":"<<i.value();
Qt API包含很多返回值为QList或QStringList的函数,要遍历这些返回的容器,必须先复制。由于Qt使用了隐式共享,这样的复制并无多大开销。例如下面的代码是正确的。
const QList<int>sizes = splitter->sizes();
QList<int>::const_iterator i;
for(i = sizes.begin();i !=sizes.end();++i)
    qDebug()<<*i;
提示:隐式共享(Implicit Sharing)是对象的管理方法。一个对象被隐式共享,只是传递该对象的一个指针给使用者,而不实际复制对象数据,只有在使用者修改数据时,才实质复制共享对象给使用者。如在上面的代码中,splitter->sizes()返回的是一个QList<int>列表对象sizes,但是实际上代码并不将splitter->sizes()表示的列表内容完全复制给变量sizes,只是传递给它一个指针。只有当sizes发生数据修改时,才会将共享对象的数据复制给sizes,这样避免了不必要的复制,减少了资源占用。
下面的代码是错误的。对于STL类型的迭代器,隐式共享还涉及另外一个问题,即当有一个迭代器在操作一个容器变量时,不要去复制这个容器变量。
QList<int>::const_iterator i;
for(i =splitter->sizes().begin();i !=splitter->sizes().end();++i)
    qDebug()<<*i;

2.3.2 STL风格迭代器的API

下面的表概括了STL风格迭代器的API:
表达式
用途
*i
返回当前项
++i
将迭代器指向下一项
i += n
迭代器向前移动n项
--i
将迭代器指向上一项
i -= n
将迭代器你向后移动n项
i - j
返回迭代器i和j之间项的数目

3.foreach关键字

如果只是想遍历容器中所有的项,可以使用foreach关键字。foreach是<QtGlobal>头文件中定义的一个宏。使用foreach的句法是:
foreach(variable,container)
使用foreach的代码比使用迭代器更简洁。例如,使用foreach遍历一的示例代码如下:
QLinkedList<QString>list;
...
QString str;
foreach(str,list)
    qDebug()<<str;
用于迭代的变量也可以在foreach语句里定义,foreach语句也可以使用花括号,可以使用break
退出迭代,示例代码如下:
QLinkedList<QString>list;
foreach(const QString &str,list)
{
    if(str.isEmpty())
        break;
    qDebug()<<str;
}
对于QMap和QHash,foreach会自动访问“键一一值”对里的值,所以无需调用values()。如果需要访问键则可以调用keys(),示例代码如下:
QMap<QString,int>map;
......
foreach(const QString &str,map.keys())
qDebug()<<str<<":"<<map.value(str);
对于多值映射,可以使用两重foreach语句,示例代码如下:
QMultiMap<QString,int>map;
...
foreach(const QString &str,map.uniqueKeys())
    foreach(int i,map.values(str))
        qDebug()<<str<<":"<<i;
注意:foreach关键字遍历一个容器变量是创建了容器的一个副本,所以不能修改原来容器变量的数据项。

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

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

相关文章

ORACLE使用Mybatis-plus批量插入

ORACLE使用mybatis-plus自带的iservice.saveBatch方法时&#xff0c;会报DML Returing cannot be batch错误&#xff1a; 推测原因是oracle不支持insert into table_name (,) values &#xff08;&#xff0c;&#xff09;,&#xff08;&#xff09;的写法。且oracle不会自动生…

华为Watch Buds手表耳机声音小怎么办?试试这些方法

使用华为 WATCH Buds 的手表耳机的铁子们&#xff0c;你们有没有和我遇到同样的问题&#xff1a;刚用的时候音质超级好&#xff0c;但是用了一段时间后&#xff0c;偶尔会遇到耳机声音变小&#xff0c;甚至出现左右耳音量不一致的情况。 如果有的话&#xff0c;不用担心&#…

【libcurl库】安装及其编程访问百度首页(一)

一、Libcurl库简介 二、Libcurl等三方库的通用编译方法 库的配置、编译、安装 &#xff08;1&#xff09;安装位置的更改&#xff08;2&#xff09;编译、安装&#xff08;3&#xff09;预览 三、调用libcurl编程访问百度主页 编译&#xff08;编译时链接库、头文件路径&am…

操作系统的特征

一、并发 并发&#xff1a;指两个或多个事件在同一时间间隔内发生。这些事件宏观上是同时发生的&#xff0c;但微观上是交替发生的。 并行 &#xff1a;是指两个或多个事件同一事件发生。 操作系统的并发性指计算机系统中“同时”运行着多个程序&#xff0c;这些程序宏观上看…

Hadoop学习笔记(HDP)-Part.20 安装Flume

目录 Part.01 关于HDP Part.02 核心组件原理 Part.03 资源规划 Part.04 基础环境配置 Part.05 Yum源配置 Part.06 安装OracleJDK Part.07 安装MySQL Part.08 部署Ambari集群 Part.09 安装OpenLDAP Part.10 创建集群 Part.11 安装Kerberos Part.12 安装HDFS Part.13 安装Ranger …

java小工具util系列3:JSON转实体类对象工具

文章目录 准备工作1.JSONObject获取所有的key2.集合中实体对象转换 list中Enrey转Dto3.字符串转List<BusyTimeIndicatorAlarmThreshold>4.json字符串转JSONObject5.list根据ids数组过滤list6.json字符串转JavaBean对象7.json对象转javabean8.jsonObject转map9.List\<U…

微服务的利与弊

一、前言 自从大多数web架构从单体演进到服务拆分&#xff0c;到微服务一统天下的几年来&#xff0c;应该没有web应用不是微服务架构的吧。最开始是阿里的doubble分层架构&#xff0c;到后来的SpringCloud全家桶&#xff0c;还有各个大厂自己定义的一套服务治理框架。微服务无…

DS二分查找_搜索二维矩阵

Description 使用二分查找法来判断m*n矩阵matrix中是否存在目标值target。 该矩阵有以下特性&#xff1a; 1. 每行中的整数从左到右升序排列&#xff1b; 2. 每行的第一个整数大于前一行的最后一个整数。 Input 第一行输入m和n&#xff0c;分别表示矩阵的行数和列数&#…

python 自动发送邮件

对于工作中&#xff0c;如果每日需要发送具有规律性的邮件&#xff0c;采用python自动发送可以节约很多时间&#xff0c;提高效率。本文主要以腾讯企业邮箱为例&#xff0c;介绍python自动发送邮件&#xff08;含附件&#xff09;的功能。为保护隐私&#xff0c;代码中的邮箱及…

Edge调用Aria2下载

一、准备工作 1、Edge浏览器&#xff1a;Windows系统自带或点击下载&#xff1b;   2、Aria2 gui&#xff1a;点击github下载或自行搜索下载其他版本&#xff1b; 二、启动Aria2 gui 解压下载的Aria2 gui到任意目录&#xff0c;点击“Aria2c启动器”或“AriaNg启动器”皆可。…

文章解读与仿真程序复现思路——中国电机工程学报EI\CSCD\北大核心《“无废”电–氢充能服务区多源微网优化运行模型》

这个文章标题涉及到“无废”电–氢充能服务区、多源微网和优化运行模型。下面我将对标题中的关键术语进行解读&#xff1a; “无废”电–氢充能服务区&#xff1a; 无废电&#xff08;Waste-free electricity&#xff09;&#xff1a; 这可能指的是通过某种方式生产的电力&…

从零开始学习 JS APL(六):完整指南和实例解析

学习目标&#xff1a; 1. 能够利用正则表达式校验输入信息的合法性 2. 具备利用正则表达式验证小兔鲜注册页面表单的能力 学习内容&#xff1a; 正则表达式 综合案例 阶段案例 学习时间&#xff1a; 周一至周五晚上 7 点—晚上9点周六上午 9 点-上午 11 点周日下午 3 点-下…

短视频系统设计:如何支持三千万用户同时在线看视频?

Java全能学习面试指南&#xff1a;https://javaxiaobear.cn 短视频&#xff08;short video&#xff09;通常时长在 15 分钟以内&#xff0c;主要是在移动智能终端上进行拍摄、美化编辑或加特效&#xff0c;并可以在网络社交平台上进行实时分享的一种新型视频形式。短视频具有时…

简单地将附件POST到Notes应用中

大家好&#xff0c;才是真的好。 这半年我们会讲很多开发知识&#xff0c;这篇紧接上篇《通过URL将HTML表单数据创建到Domino应用中》&#xff0c;讲述如何将附件POST到Note应用中。 原理是一样&#xff0c;首先&#xff0c;你得有一个表单&#xff0c;我们用html写一个&…

Android 缩减、混淆处理和优化应用

为了尽可能减小应用的大小&#xff0c;您应在发布 build 中启用缩减功能来移除不使用的代码和资源。启用缩减功能后&#xff0c;您还会受益于两项功能&#xff0c;一项是混淆处理功能&#xff0c;该功能会缩短应用的类和成员的名称&#xff1b;另一项是优化功能&#xff0c;该功…

云HIS:新一代云架构医院信息管理系统源码(java语言)

云HIS信息管理云平台&#xff0c;提供全方位的临床系统应用&#xff0c;是国内领先的以云计算为基础&#xff0c;以云计算赋能医疗机构&#xff0c;是颠覆传统医疗信息化业态的技术与模式创新&#xff0c;以SaaS方式&#xff0c;为医疗机构提供信息系统服务&#xff0c;满足从医…

Deep Learning(wu--84)调参、正则化、优化--改进深度神经网络

文章目录 2偏差和方差正则化梯度消失\爆炸权重初始化导数计算梯度检验OptimizationMini-Batch 梯度下降法指数加权平均偏差修正RMSpropAdam学习率衰减局部最优问题 调参BNsoftmax framework 2 偏差和方差 唔&#xff0c;这部分在机器学习里讲的更好点 训练集误差大&#xff…

【生信分析】基因组学导论

由于工作需要&#xff0c;现在开始跨行学生信&#xff01; 祝我成功 目标&#xff1a;通过一周的学习能对对不同高通量测序数据集&#xff08;RNA-seq、ChIP-seq、BS-seq 和多组学集成&#xff09;进行分析。 配置环境 if (!requireNamespace("BiocManager", quiet…

C语言能判断一个变量是int还是float吗?

C语言能判断一个变量是int还是float吗&#xff1f; 在开始前我有一些资料&#xff0c;是我根据自己从业十年经验&#xff0c;熬夜搞了几个通宵&#xff0c;精心整理了一份「C语言从专业入门到高级教程工具包」&#xff0c;点个关注&#xff0c;全部无偿共享给大家&#xff01;&…

初识消息队列

1、消息 消息&#xff08;Message&#xff09;是指在应用间传送的数据。消息可以非常简单&#xff0c;比如只包含文本字符串&#xff0c;也可以更复杂&#xff0c;可能包含嵌入对象。 2、消息队列 消息队列&#xff08;Message Queue&#xff09;是一种应用间的通信方式&#…