【Qt开发流程】之容器类1:介绍及常用容器类和使用Java风格迭代器进行遍历

概述

Qt库提供了一组通用的基于模板的容器类。这些类可用于存储指定类型的项。例如,如果需要一个可调整大小的QString数组,可以使用QVector<QString>

这些容器类被设计成比STL容器更轻、更安全、更易于使用。如果不熟悉STL,或者更喜欢用Qt方式进行变成,那么就可以使用这些类来代替STL类

容器类是隐式共享的,它们是可重入的,并且它们针对速度、低内存消耗和最小的内联代码扩展进行了优化,从而产生更小的可执行文件。此外,当所有访问它们的线程都将它们用作只读容器时,它们是线程安全的。

要遍历存储在容器中的项,可以使用两种类型的迭代器之一:java风格的迭代器stl风格的迭代器java风格的迭代器更容易使用并提供高级功能,而STL风格的迭代器稍微更有效,可以与QtSTL的泛型算法一起使用。

Qt还提供了foreach关键字,它可以很容易地遍历容器中存储的所有项。

Qt容器类介绍

Qt提供了以下顺序容器:QList, QLinkedList, QVector, QStackQQueue。对于大多数应用程序,QList是最好的类型。虽然它是作为数组列表实现的,但它提供了非常快的前置和追加。如果需要一个链表,使用QLinkedList;如果想让项目占用连续的内存位置,使用QVectorQStackQQueue是提供后进先出先进先出语义的方便类。

Qt还提供了这些关联容器:QMapQMultiMapQHashQMultiHashQSet“Multi”容器方便地支持与单个键关联的多个值。“哈希”容器通过使用哈希函数而不是对排序集进行二进制搜索来提供更快的查找。

作为特殊情况,QCacheqconsecuousscache类在有限的缓存存储中提供了有效的对象哈希查找。

以下是常用容器类:

介绍
QList这是目前为止最常用的容器类。它存储可以通过索引访问的给定类型(T)的值列表。在内部,QList使用数组实现,确保基于索引的访问非常快。
QLinkedList可以使用QList::append()和QList::prepend()在列表的两端添加项,也可以使用QList::insert()在中间插入项。与任何其他容器类不同,QList经过了高度优化,可以在可执行文件中扩展到尽可能少的代码。QStringList继承自QList。这类似于QList,不同之处在于它使用迭代器而不是整数索引来访问项。在大列表中插入时,它还提供了比QList更好的性能,并且具有更好的迭代器语义。(只要QLinkedList中的元素存在,指向该元素的迭代器就保持有效,而指向QList的迭代器在插入或删除后就会失效。)
QVector这将给定类型的值数组存储在内存中的相邻位置。在vector的前面或中间插入可能会非常慢,因为它可能导致大量项必须在内存中移动一个位置。
QStack这是QVector的一个方便子类,提供“后进先出”(LIFO)语义。它为QVector中已经存在的函数添加了以下函数:push()、pop()和top()。
QQueue这是QList的一个方便子类,提供“先进先出”(FIFO)语义。它向QList中已经存在的函数添加了以下函数:enqueue()、dequeue()和head()。
QSet这提供了一个具有快速查找功能的单值数学集。
QMap<Key, T>它提供了一个字典(关联数组),将Key类型的键映射到t类型的值。通常每个键都与单个值相关联。QMap按Key顺序存储数据;如果顺序不重要,QHash是一个更快的选择。
QMultiMap<Key, T>这是QMap的一个方便的子类,它为多值映射提供了一个很好的接口,即一个键可以与多个值相关联的映射。
QHash<Key, T>它具有与QMap几乎相同的API,但提供了明显更快的查找。QHash以任意顺序存储数据。
QMultiHash<Key, T>这是QHash的一个方便的子类,它为多值哈希提供了一个很好的接口。


容器是可以嵌套。例如,完全可以使用QMap<QString, QList<int>>,其中键类型是QString,值类型是QList<int>

容器在单独的头文件中定义,其名称与容器相同(例如,<QLinkedList>)。为方便起见,在<QtContainerFwd>中向前声明了容器。

存储在各种容器中的值可以是任何可赋值的数据类型。要符合条件,类型必须提供默认构造函数、复制构造函数和赋值操作符。这涵盖了可能想要存储在容器中的大多数数据类型,包括基本类型,如intdouble,指针类型,以及Qt数据类型,如QString, QDateQTime,但它不包括QObject或任何QObject子类(QWidget, QDialog, QTimer等)。如果尝试实例化QList<QWidget>,编译器将报错QWidget的复制构造函数和赋值操作符被禁用。如果希望将这些类型的对象存储在容器中,请将它们存储为指针,例如QList<QWidget *>

下面是一个满足可赋值数据类型要求的自定义数据类型示例:

  class Employee{public:Employee() {}Employee(const Employee &other);Employee &operator=(const Employee &other);private:QString myName;QDate myDateOfBirth;};


如果没有提供复制构造函数或赋值操作符,c++提供了执行逐个成员复制的默认实现。在上面的例子中,这就足够了。此外,如果不提供任何构造函数,c++还提供一个默认构造函数,该构造函数使用默认构造函数初始化其成员。尽管它没有提供任何显式构造函数或赋值操作符,但下列数据类型可以存储在容器中:

  struct Movie{int id;QString title;QDate releaseDate;};


有些容器对它们可以存储的数据类型有额外的要求。例如,QMap<Key, T>Key类型必须提供操作符<()。这些特殊需求在类的详细描述中都有文档。在某些情况下,特定功能有特殊要求;这些是按功能描述的。如果不满足要求,编译器总是会发出一个错误。

Qt的容器提供了operator<<()operator>>(),因此它们可以很容易地使用QDataStream进行读写。这意味着存储在容器中的数据类型还必须支持operator<<()operator>>()。提供这样的支持是直截了当的;下面是我们如何为上面的Movie结构体做到这一点:

  QDataStream &operator<<(QDataStream &out, const Movie &movie){out << (quint32)movie.id << movie.title<< movie.releaseDate;return out;}QDataStream &operator>>(QDataStream &in, Movie &movie){quint32 id;QDate date;in >> id >> movie.title >> date;movie.id = (int)id;movie.releaseDate = date;return in;}


某些容器类函数的文档引用默认构造的值;例如,QVector用默认构造的值自动初始化它的项,如果指定的键不在映射中,QMap::value()返回一个默认构造的值。对于大多数值类型,这仅仅意味着使用默认构造函数创建值(例如,QString的空字符串)。但是对于intdouble这样的基本类型,以及指针类型,c++语言没有指定任何初始化;在这种情况下,Qt的容器会自动将该值初始化为0

示例

以下是使用QListQMap进行介绍,其他几个容器可以参考这两个进行操作,因为他们的API很相似。
QList
QList是Qt中提供的一个容器类,它是一个动态数组,可以自动调整大小以容纳新元素

  • append:在列表的末尾添加一个新元素。
QList<int> list;
list.append(1);
list.append(2);
list.append(3);
// list: {1, 2, 3}
  • prepend:在列表的开头添加一个新元素。
QList<int> list;
list.prepend(1);
list.prepend(2);
list.prepend(3);
// list: {3, 2, 1}
  • insert:在列表的指定位置插入一个新元素。
QList<int> list;
list << 1 << 2 << 4;
list.insert(2, 3);
// list: {1, 2, 3, 4}
  • replace:用新元素替换列表中的一个元素。
QList<int> list;
list << 1 << 2 << 3;
list.replace(1, 4);
// list: {1, 4, 3}
  • removeAt:从列表中删除指定索引处的元素。
QList<int> list;
list << 1 << 2 << 3;
list.removeAt(1);
// list: {1, 3}
  • swap:交换列表中两个元素的位置。
QList<int> list;
list << 1 << 2 << 3;
list.swap(0, 2);
// list: {3, 2, 1}
  • contains:检查列表中是否包含某个元素。
QList<int> list;
list << 1 << 2 << 3;
if (list.contains(2)) {qDebug() << "2 is in the list.";
}
  • count:返回列表中某个元素的数量。
QList<int> list;
list << 1 << 2 << 2 << 3 << 2;
int count = list.count(2); // count is 3
  • indexOf:返回列表中某个元素的第一个索引。
QList<int> list;
list << 1 << 2 << 2 << 3 << 2;
int index = list.indexOf(2); // index is 1
  • lastIndexOf:返回列表中某个元素的最后一个索引。
QList<int> list;
list << 1 << 2 << 2 << 3 << 2;
int index = list.lastIndexOf(2); // index is 4
  • empty:检查列表是否为空。
QList<int> list;
if (list.empty()) {qDebug() << "The list is empty.";
}
  • size:返回列表的大小。
QList<int> list;
list << 1 << 2 << 3;
int size = list.size(); // size is 3
  • clear:删除列表中的所有元素。
QList<int> list;
list << 1 << 2 << 3;
list.clear();
// list is now empty

对于只读的访问,一般使用at()函数,因为比"[ ]"操作符快很多。
QMap
QMap是Qt中提供的一个容器类,它是一个关联数组,可以将键值对存储在一起.

  • insert:在映射中插入一个键值对。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
map.insert("Charlie", 35);
// map: { "Alice": 30, "Bob": 25, "Charlie": 35 }
  • replace:用新值替换映射中的一个值。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
map.insert("Charlie", 35);
map.replace("Bob", 20);
// map: { "Alice": 30, "Bob": 20, "Charlie": 35 }
  • remove:从映射中删除指定键的值。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
map.insert("Charlie", 35);
map.remove("Bob");
// map: { "Alice": 30, "Charlie": 35 }
  • swap:交换两个映射的内容。
QMap<QString, int> map1, map2;
map1.insert("Bob", 25);
map1.insert("Alice", 30);
map2.insert("Charlie", 35);
map2.insert("Dave", 40);
map1.swap(map2);
// map1: { "Charlie": 35, "Dave": 40 }
// map2: { "Alice": 30, "Bob": 25 }
  • contains:检查映射中是否包含指定键。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
if (map.contains("Bob")) {qDebug() << "Bob is in the map.";
}
  • count:返回映射中指定键的数量。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
map.insert("Charlie", 35);
int count = map.count("Bob"); // count is 1
  • value:返回映射中指定键的值,如果键不存在则返回默认值。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
int value = map.value("Charlie", 0); // value is 0
  • keys:返回映射中所有的键。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
QList<QString> keys = map.keys(); // keys: { "Bob", "Alice" }
  • values:返回映射中所有的值。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
QList<int> values = map.values(); // values: { 25, 30 }
  • empty:检查映射是否为空。
QMap<QString, int> map;
if (map.empty()) {qDebug() << "The map is empty.";
}
  • size:返回映射中键值对的数量。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
int size = map.size(); // size is 2
  • clear:删除映射中的所有键值对。
QMap<QString, int> map;
map.insert("Bob", 25);
map.insert("Alice", 30);
map.clear();
// map is now empty

遍历容器

要遍历存储在容器中的项,可以使用两种类型的迭代器之一:java风格的迭代器stl风格的迭代器java风格的迭代器更容易使用并提供高级功能,而STL风格的迭代器稍微更有效,可以与QtSTL的泛型算法一起使用。

Qt还提供了foreach关键字,它可以很容易地遍历容器中存储的所有项。
在这里插入图片描述
在本文档中,将集中讨论QList和QMap。QLinkedList、QVector和QSet的迭代器类型与QList的迭代器具有完全相同的接口;类似地,QHash的迭代器类型与QMap的迭代器具有相同的接口。
与stl风格的迭代器(将在下面介绍)不同,java风格的迭代器指向项目之间,而不是直接指向项目。由于这个原因,它们要么指向容器的最开始(在第一个项目之前),要么指向容器的最末尾(在最后一个项目之后),要么指向两个项目之间。下图用红色箭头显示了包含四个元素的列表的有效迭代器位置:
在这里插入图片描述
下面,按顺序遍历QList中的所有元素,并将它们打印到控制台:

  QList<QString> list;list << "A" << "B" << "C" << "D";QListIterator<QString> i(list);while (i.hasNext())qDebug() << i.next();

它的工作原理如下:将要迭代的QList传递给QListIterator构造函数。此时,迭代器位于列表中第一项的前面(在项“A”之前)。然后调用hasNext()来检查迭代器后是否有项。如果有,则调用next()跳过该项。next()函数返回它跳过的项。对于QList,该项的类型为QString。
下面是如何在QList中向后迭代:

  QListIterator<QString> i(list);i.toBack();while (i.hasPrevious())qDebug() << i.previous();

代码与向前迭代是对称的,除了我们首先调用toBack()将迭代器移动到列表中的最后一项之后。
下图说明了在迭代器上调用next()和previous()的效果:
在这里插入图片描述
QListIterator常用API:

方法行为
toFront()将迭代器移动到列表的前面(在第一项之前)
toBack()将迭代器移动到列表的后面(在最后一项之后)
hasNext()如果迭代器不在列表的末尾,则返回true
next()返回下一项并将迭代器向前移动一个位置
peekNext()不移动迭代器返回下一项
hasPrevious()如果迭代器不在列表的前面,则返回true
previous()返回前一项并将迭代器向后移动一个位置
peekPrevious()不移动迭代器返回前一项

QListIterator不提供在迭代时从列表中插入或删除项的函数。要做到这一点,必须使用QMutableListIterator
下面是一个使用QMutableListIteratorQList<int>中删除所有奇数的例子:

  QMutableListIterator<int> i(list);while (i.hasNext()) {if (i.next() % 2 != 0)i.remove();}

循环中的next()调用每次都进行。它跳过列表中的下一项。remove()函数删除了我们从列表中跳过的最后一项。调用remove()不会使迭代器失效,所以继续使用它是安全的。这在向后迭代时同样有效:

  QMutableListIterator<int> i(list);i.toBack();while (i.hasPrevious()) {if (i.previous() % 2 != 0)i.remove();}

就像remove()一样,setValue()对我们跳过的最后一项进行操作。如果向前迭代,这是迭代器前面的项;如果向后迭代,这是迭代器后面的项。
next()函数返回对列表中项目的非const引用。对于简单的操作,我们甚至不需要setValue():

  QMutableListIterator<int> i(list);while (i.hasNext())i.next() *= 2;

如上所述,QLinkedList、QVector和QSet的迭代器类具有与QList完全相同的API。现在我们将转向QMapIterator,它有点不同,因为它对(键,值)对进行迭代。
与QListIterator一样,QMapIterator提供toFront()、toBack()、hasNext()、next()、peekNext()、hasPrevious()、previous()和peekPrevious()。键和值组件是通过对next()、peekNext()、previous()或peekPrevious()返回的对象调用key()和value()来提取的。
下面的例子删除所有以"City"结尾的(capital, country)对:

  QMap<QString, QString> map;map.insert("Paris", "France");map.insert("Guatemala City", "Guatemala");map.insert("Mexico City", "Mexico");map.insert("Moscow", "Russia");...QMutableMapIterator<QString, QString> i(map);while (i.hasNext()) {if (i.next().key().endsWith("City"))i.remove();}

QMapIterator还提供了一个key()和一个value()函数,它们直接对迭代器进行操作,并返回迭代器跳到上面的最后一项的键和值。例如,下面的代码将QMap的内容复制到QHash中:

  QMap<int, QWidget *> map;QHash<int, QWidget *> hash;QMapIterator<int, QWidget *> i(map);while (i.hasNext()) {i.next();hash.insert(i.key(), i.value());}

如果要遍历具有相同值的所有项,可以使用findNext()或findPrevious()。下面是一个例子,删除所有具有特定值的项:

  QMutableMapIterator<int, QWidget *> i(map);while (i.findNext(widget))i.remove();

结论

想改变物质生活,后来发现,改变心态更容易

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

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

相关文章

低多边形3D建模石头材质纹理贴图

在线工具推荐&#xff1a; 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 当谈到游戏角色的3D模型风格时&#xff0c;有几种不同的风格&#xf…

云计算在计算机领域的应用与发展

云计算在计算机领域的应用与发展 一、引言 随着科技的不断发展&#xff0c;计算机领域已经成为当今社会最为活跃和创新的领域之一。云计算作为一种新兴的计算模式&#xff0c;已经在计算机领域中得到了广泛的应用&#xff0c;并且正在不断地推动着计算机领域的发展。本文将探…

Ultimate VFX

Ultimate VFX 构建套件:

【利用二手车数据进行可视化分析】

利用二手车数据进行可视化分析 查看原始数据去除重复数据需求分析1.统计全国总共有多少量二手车&#xff0c;用KPI图进行展示2.统计安徽总共有多少量二手车&#xff0c;用KPI图进行展示3.统计合肥总共有多少量二手车&#xff0c;用KPI图进行展示4.取最贵的10辆二手车信息&#…

web,Apache简述

一.HTTP请求访问的完整过程 1.建立连接 2.接收请求 3.处理请求 4.访问资源 服务器获取请求报文中请求的资源web服务器&#xff0c;即存放了web资源的服务器&#xff0c;负责向请求者提供对方请求的静态资源&#xff0c;或动态运行后生成的资源 静态资源&#xff1a;不需要…

C语言 内联函数 + 递归函数

函数分类 内联函数 1&#xff09;内联函数在编译时将函数的代码直接插入到调用它的地方&#xff0c;而不是通过函数调用的方式执行&#xff0c;从而减少了函数调用的开销&#xff0c;提高了代码的执行速度 2&#xff09;使用 inline 关键字来声明 3&#xff09;将函数声明为内联…

小目标检测模型设计的一点思考

1. 小目标的特性 目标之间的交叠概率比较低&#xff0c;即使有交叠&#xff0c;其IoU多数情况下也是比较小的 AI-TOD Tiny Person Dateset 小目标自身的纹理显著度有强弱区别&#xff0c;但是总体来说纹理特征都较弱&#xff0c;很多时候需要借助一定的图像上下文来帮助确认 …

Java解决岛屿周长问题

Java解决岛屿周长问题 01 题目 给定一个 row x col 的二维网格地图 grid &#xff0c;其中&#xff1a;grid[i][j] 1 表示陆地&#xff0c; grid[i][j] 0 表示水域。 网格中的格子 水平和垂直 方向相连&#xff08;对角线方向不相连&#xff09;。整个网格被水完全包围&am…

ssm校园论坛管理系统项目分享

校园论坛管理系统是基于java编程语言&#xff0c;mysql数据库&#xff0c;ssm框架和idea工具开发&#xff0c;本系统主要分为学生用户&#xff0c;管理员两个角色&#xff0c;其中用户可以注册登陆系统&#xff0c;在线发帖&#xff0c;查看栏目帖子&#xff0c;回复帖子&#…

某音上很火的圣诞树分享

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 效果截图&#xff08;这里不给动态了&#xff0c;某音到处都是了&#xff09;&#xff1a; 源代码&#xff1a; <script src"…

Spring Boot 3 集成 MyBatis详解

MyBatis是一款开源的持久层框架&#xff0c;它极大地简化了与数据库的交互流程。与类似Hibernate的ORM框架不同&#xff0c;MyBatis更具灵活性&#xff0c;允许开发者直接使用SQL语句与数据库进行交互。Spring Boot和MyBatis分别是两个功能强大的框架&#xff0c;它们的协同使用…

Linux shell编程学习笔记34:eval 命令

0 前言 在JavaScript语言中&#xff0c;有一个很特别的函数eval&#xff0c;eval函数可以将字符串当做 JavaScript 代码执行&#xff0c;返回表达式或值。 在Linux Shell 中也提供了内建命令eval&#xff0c;它是否具有JavaScript语言中eval函数的功能呢&#xff1f; 1 eval命…

GPIO的使用--USART串口通信--传感器控制数据

目录 一、串口通信 1、概念 2、原理图 3、使用步骤 &#xff08;1&#xff09;寻找串口位置 &#xff08;2&#xff09;确定引脚编号 &#xff08;3&#xff09;编写代码 4、实验结果 实验代码 main.c usart.c usart.h 一、串口通信 1、概念 串行接口是一种可以将…

DiffiT

本文首发于AIWalker&#xff0c;欢迎关注。 https://arxiv.org/abs/2312.02139 https://github.com/NVlabs/DiffiT 扩散模型以其强大的表达能力和高样本质量在许多领域得到了新的应用。对于样本生成&#xff0c;这些模型依赖于通过迭代去噪生成图像的去噪神经网络。然而&#x…

图像的均方差和信噪比计算

图像的均方差和信噪比计算 一、均方差1、公式2、代码 二、信噪比1、公式2、代码 图像的均方差和信噪比公式及代码&#xff0c;代码基于opencv和C实现。 一、均方差 均方误差&#xff0c;英文简称&#xff1a;MSE&#xff0c;英文全称&#xff1a;“Mean Square Error”。 衡量…

接口测试-Jmeter使用

一、线程组 1.1 作用 线程组就是控制Jmeter用于执行测试的一组用户 1.2 位置 右键点击‘测试计划’-->添加-->线程(用户)-->线程组 1.3 特点 模拟多人操作线程组可以添加多个&#xff0c;多个线程组可以并行或者串行取样器(请求)和逻辑控制器必须依赖线程组才能…

「Verilog学习笔记」多bit MUX同步器

专栏前言 本专栏的内容主要是记录本人学习Verilog过程中的一些知识点&#xff0c;刷题网站用的是牛客网 输入数据暂存在data_reg中&#xff0c;使能信号data_en用打两拍的方式跨时钟域传输到时钟域B&#xff0c;最后data_out根据使能信号更新数据。data_en信号在A时钟域用一个D…

Redis | Redis入门学习介绍及常见原理剖析

关注wx&#xff1a;CodingTechWork Redis介绍 概述 Redis是NoSQL&#xff0c;是key-value分布式内存数据库。 缓存 缓存是将数据从慢的介质换到快的介质上&#xff0c;提高读写效率和性能&#xff0c;并降低数据库的读写成本。内存的速度一般都远远大于硬盘的速度&#xf…

三个臭皮匠(ctr,nerdctl,crictl)顶一个诸葛亮(docker)

文章目录 containerd简介 nerdctl简介安装精简 Minimal 安装完整Full 安装启动服务 命令参数容器运行容器列出容器详情容器日志容器进入容器停止容器删除镜像列表镜像拉取镜像标签镜像导出镜像导入镜像删除镜像构建配置tab键配置加速配置仓库http方式https方式 ctr简介命令参数…

AMC8美国数学竞赛历年真题集在线练习操作指南和2024年备考建议

今天是2023年12月10日&#xff0c;距离2024年的AMC8美国数学竞赛的举办还有40天时间。据六分成长了解&#xff0c;有一些孩子报名参加了AMC8的机构培训班系统学习&#xff0c;也有一些孩子选择了自己自学备考。 有家长问AMC8的培训是否一定要参加机构的培训班学习&#xff1f;…