IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> foreach、qAsConst用法总结 -> 正文阅读

[C++知识库]foreach、qAsConst用法总结

预备知识

本文要用到Qt的detach、隐式共享技术,如果对这两种技术不了解的读者,请在qt官方自带的Assistant 中的“索引”tab页的搜索框中输入implicitly shared,就可以看到Qt官方的阐述。

foreach说明

foreach关键字是Qt中用于遍历容器的一个关键字,是Qt官方自己实现的,其不是C++标准中存在的关键字。其语法如下:

foreach (variable, container)

利用foreach可以对Qt自己的容器如:QVector、QMap、 QHash、QLinkedList、QList进行遍历,如下:


  QLinkedList<QString> list;
  ...
  foreach (const QString &str, list) {
      if (str.isEmpty())
          break;
      qDebug() << str;
  }

对于QMap、 QHash键值对类型遍历如下:

    QMap<int, QString> mpStudent;
    mpStudent[0] = "dan";
    mpStudent[1] = "shi";
    mpStudent[2] = "ming";

	foreach(auto var, mpStudent)
	{
		qDebug() << var<< "\r\n";
	}

结果如下:

可以看到,将foreach运用于QMap、QHash遍历时,默认返回QMap、QHash键值对的值部分。如果要遍历QMap、QHash的键部分,可将foreach语句改为如下:

foreach(auto var, mpStudent.keys())

如果要遍历键又要遍历值,则请用迭代器。

foreach也可以遍历STL的容器,如下:

    vector<int>vtSTLTest;

	for (auto i = 0; i < 10; ++i)
	{
		vtSTLTest.push_back(i);
	}
	foreach(auto var, vtSTLTest)
	{
		qDebug() << var << "\r\n";
	}

Qt进入到foreach循环后,会将容器自动拷贝一份,因为Qt的容器都是隐式共享的(类似于智能指针),所以拷贝Qt自己的容器过程非常快,几乎对性能没啥大的影响,但因为STL的容器没有像Qt自己的容器那样实现隐式共享,所以如果拷贝的是STL容器,有时代价是昂贵的。因为foreach自动拷贝了原来的容器,所以在foreach循环内,对容器的更改只会影响到拷贝的容器,对于原始容器则不会有影响,基于这一点,在foreach语法中,variable不能为非常量的引用,只能为值传递或常量引用,否则就可以对原始容器进行更改了,因此下面的语法,编译器都会报错

   vector<int>vtSTLTest;

	for (auto i = 0; i < 10; ++i)
	{
		vtSTLTest.push_back(i);
	}
	foreach(auto& var, vtSTLTest)
	{
		var[1] = 2;
		qDebug() << var << "\r\n";
	}

??试图修改原始容器索引为1的元素,会报错

    vector<int>vtSTLTest;

	for (auto i = 0; i < 10; ++i)
	{
		vtSTLTest.push_back(i);
	}
	foreach(int& var, vtSTLTest)
	{
		qDebug() << var << "\r\n";
	}

?非常量的int引用会导致原始容器被修改,所以会报错。

foreach(int& var, vtSTLTest)

将上面一句代码改为:

foreach(const int& var, vtSTLTest)

或改为:

foreach(int var, vtSTLTest)

不会报错,这种情况下,不会导致对原始容器更改。

如果在用foreach遍历容器时,外部更改了原始容器,则它不会影响foreach循环,即foreach循环输出的依然是原始容器未修改的值,这证明foreach确实复制了原始容器,操作的是复制容器而不是原始容器。如下:

#include "QtWidgetsApplication2.h"
#include <QVector>
#include <map>
#include <thread>
using namespace  std;
 
void fun(QVector<int>& vtQTest)
{
	for (auto i = 0; i < 10; ++i)
	{
		vtQTest[i] = i+2;
	}
}

QtWidgetsApplication2::QtWidgetsApplication2(QWidget *parent)
    : QWidget(parent)
{
    ui.setupUi(this);

	QVector<int>vtQTest;
	for (auto i = 0; i < 10; ++i)
	{
		vtQTest.push_back(i);
	}

    std::thread* p{nullptr};
    foreach ( int var ,vtQTest)
    {
        if (nullptr == p)
        {
            p = new std::thread(fun, std::ref(vtQTest));

            // 等待5秒,以便原始容器内的值能被线程全部更改完
            std::this_thread::sleep_for(std::chrono::milliseconds(5000)); 
        }
 
        // 虽然上面线程修改了原始容器,但这里依然输出的是未修改的值
        qDebug() << var << "\r\n";
    }


	foreach(int var, vtQTest)  
	{
	    // 进入到这个foreach,会重新拷贝一份vtQTest,而vtQTest在上面的线程中
        // 被更改了,所以这里输出的更改后的值。
		qDebug() << "new :" << var << "\r\n";
	}
   
}

输出如下:

?

基于范围的循环可能导致Qt库容器会执行detach操作,如下:


      QString s = ...;
      for (QChar ch : s) // detaches 's' (performs a deep-copy if 's' was shared)
          process(ch);

上述代码是用for实现的基于范围的循环,符合C++11标准语法。当process函数对ch进行更改操作之前,s就会被detach,之后再深拷贝出一个和s一样的对象,而有时候我们不想被detach和深拷贝,这样在原本可以浅拷贝的情况下执行深拷贝,同样影响效率,代价高昂。但换成如下的foreach,则s不会被detach,仅仅进行指针级的浅拷贝:

      QString s = ...;
      foreach (QChar ch, s)  
          process(ch);

但是在STL的容器上用foreach,却会导致复制操作,前文已经说过,STL的容器没有实现隐式共享,这有时会导致复制的代价很高,对性能和效率有影响。为了兼顾这两者,Qt官方给出了如下建议:

  • 对于Qt自己实现的容器,如:QVector、QMap、 QHash、QLinkedList、QList等,建议用foreach进行循环。
  • 对于STL的容器,建议用for(var : container)基于范围的循环。

qAsConst?

? ?自Qt 5.7版本以来,引入了qAsConst函数,该函数Qt官方的解释如下:

  • This function is a Qt implementation of C++17's std::as_const(),this function turns non-const lvalues into const lvalues
  • Added qAsConst function to help using non-const Qt containers in C++11 range for loops
  • Its main use in Qt is to prevent implicitly-shared Qt containers from detaching

意思就是说:

  • 这个函数实现了C++17标准中的std::as_const()函数的功能,将一个非常量的左值转为常量的左值。
  • 增加qAsConst函数是为了Qt自己的容器能实现C++11标准的基于范围的循环。
  • 该函数主要用于qt容器在隐式共享中不被detach。

从上文的描述,我们知道下面的代码:


      QString s = ...;
      for (QChar ch : s) // detaches 's' (performs a deep-copy if 's' was shared)
          process(ch);

将会导致s被detach,继而再执行深拷贝,而下面的代码s不会被detach,当然也就不会再执行深拷贝了

 for (QChar ch : qAsConst(s)) // ok, no detach attempt
          process(ch);

当然,在这种情况下,你也许会说,像下面那样将s声明为const,也不会执被detach:


      const QString s = ...;
      for (QChar ch : s) // ok, no detach attempt on const objects
          process(ch);

但是在编程时、在现实中,声明为const往往不容易做到。

总结:对Qt自己实现的容器如:QVector、QMap、 QHash、QLinkedList、QList等,如果一定要用基于for(var : container)范围的循环,则请用如下形式:

for(var : qAsConst(container))

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-04-04 11:50:13  更:2022-04-04 11:52:08 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 0:49:38-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码