foreach、qAsConst用法总结

本文详细介绍了Qt中的foreach关键字及其在遍历容器如QVector、QMap、QHash等时的使用,着重讲解了隐式共享和detach的概念,以及如何避免在foreach循环中对原始容器的意外修改。同时提到了qAsConst函数的应用,以优化基于范围的循环性能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

预备知识

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

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大量数据结构采取隐式共享、写时复制,for()操作非const容器时,循环内部可能会有一些操作导致detach发生,一旦detach发生,就会导致复制整个容器。如下基于范围的循环可能导致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自己的非const 的容器能实现C++11标准的基于范围的循环。
  • 该函数主要用于qt容器在隐式共享中不被detach。

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


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

当s被修改时,将会导致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))

### QFile 文件过滤 Qt C++ 实现 在Qt框架中,`QFile`本身并不直接提供文件过滤功能。然而,可以结合其他类来实现这一目的。通常情况下,文件过滤是指基于特定条件筛选文件列表中的项目。为了实现这一点,可以使用 `QDir` 或者更高级别的 `QFileDialog` 配合正则表达式或其他逻辑来进行匹配。 #### 使用 QDir 进行简单模式匹配 对于简单的通配符风格的过滤器(例如查找所有 `.txt` 文件),可以直接使用 `QDir::entryList()` 方法并传入相应的名称过滤器: ```cpp QString path = "/path/to/directory"; QDir dir(path); QStringList filters; filters << "*.txt"; // 只显示 .txt 结尾的文件 QStringList files = dir.entryList(filters, QDir::Files); for (const QString &file : qAsConst(files)) { qDebug() << "Found file:" << file; } ``` 这种方法适用于基本的需求场景[^3]。 #### 利用 QRegularExpression 提供复杂模式支持 如果需要更加复杂的过滤规则,则可以通过遍历目录下的所有条目并与给定的正则表达式进行比较: ```cpp #include <QDir> #include <QFileInfo> #include <QRegularExpression> void filterFilesByRegex(const QString& directoryPath, const QString& pattern) { QDir dir(directoryPath); QStringList allEntries = dir.entryList(QDir::AllEntries | QDir::NoDotAndDotDot); foreach(QString entryName, allEntries){ QFileInfo fileInfo(dir.absoluteFilePath(entryName)); if(fileInfo.isFile()){ QRegularExpression re(pattern); QRegularExpressionMatch match = re.match(entryName); if(match.hasMatch()) qDebug()<<"Filtered File:"<<entryName; } } } // 调用示例 filterFilesByRegex("/path/to/search", "^.*\\.(jpg|png)$"); // 查找图片文件 ``` 上述代码展示了如何通过自定义正则表达式的模式字符串来精确控制哪些类型的文件应该被选中[^1]。 #### 整合到 GUI 应用程序中 当构建图形界面应用程序时,可能希望让用户能够交互式地选择要应用的过滤条件。此时可考虑采用 `QFileDialog` 的静态方法获取用户输入的同时设置好合适的过滤选项: ```cpp QString fileName = QFileDialog::getOpenFileName( this, tr("Select Image"), "", tr("Images (*.png *.xpm *.jpg)") // 设置允许打开的文件类型 ); if (!fileName.isEmpty()) { QImage image(fileName); ui->label->setPixmap(QPixmap::fromImage(image).scaled(ui->label->size())); } ``` 这种方式不仅简化了用户的操作流程,同时也确保了只有符合条件的文件才会呈现出来供进一步处理[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值