3

我的项目是创建一个小程序来演示搜索引擎的工作:为任意查询建立索引和返回结果。我已经完成了索引器部分的工作,现在我想通过一次索引多个文件来改进它。MainWindow 类在这里:

class MainWindow : public QMainWindow
{
    Q_OBJECT
    .....
private:
    Indexer * indexer;
    QStringList fileList;
    ....
    void index(QStringList list);
    void add(const QString &filename);
}

这是addadd需要访问fileList以避免再次索引相同的文件,因此它不能是静态方法)的实现:

void MainWindow::add(const QString &filename)
{
    if (!fileList.contains(filename))
    {
        indexer->addDocument(filename.toStdString());
        fileList.append(filename);
        qDebug() << "Indexed" << filename;
        emit updatedList(fileList);
    }
}

方法的实现index是接收一个文件列表并调用add每个文件名:

void MainWindow::index(QStringList list)
{
    ....
    QtConcurrent::map(list, &MainWindow::add);
    ....
}

编译这些代码时收到的错误是:

usr/include/qt4/QtCore/qtconcurrentmapkernel.h: In member function 'bool QtConcurrent::MapKernel<Iterator, MapFunctor>::runIteration(Iterator, int, void*) [with Iterator = QList<QString>::iterator, MapFunctor = QtConcurrent::MemberFunctionWrapper1<void, MainWindow, const QString&>]':
../search-engine/mainwindow.cpp:361:1:   instantiated from here
/usr/include/qt4/QtCore/qtconcurrentmapkernel.h:73:9: error: no match for call to '(QtConcurrent::MemberFunctionWrapper1<void, MainWindow, const QString&>) (QString&)'
/usr/include/qt4/QtCore/qtconcurrentfunctionwrappers.h:128:7: note: candidate is:
/usr/include/qt4/QtCore/qtconcurrentfunctionwrappers.h:138:14: note: T QtConcurrent::MemberFunctionWrapper1<T, C, U>::operator()(C&, U) [with T = void, C = MainWindow, U = const QString&]
/usr/include/qt4/QtCore/qtconcurrentfunctionwrappers.h:138:14: note:   candidate expects 2 arguments, 1 provided

我不太熟悉 QtConcurrent 的工作原理,文档也没有提供太多关于它的细节。我真的希望这里有人可以提供帮助。提前致谢。

4

1 回答 1

8

为了能够调用指向成员的指针,除了函数形式参数之外,您还需要该类的实例(this您在成员函数中获得的指针)。

有两种方法可以处理这个问题:创建一个简单的函子来包装调用,或者使用 lambda。

仿函数看起来像这样:

struct AddWrapper {
  MainWindow *instance;
  AddWrapper(MainWindow *w): instance(w) {}
  void operator()(QString const& data) {
    instance->add(data);
  }
};

你会像这样使用它:

AddWrapper wrap(this);
QtConcurrent::map(list, wrap);

(但请注意该包装器的生命周期。您可以使其更通用 - 例如,您还可以在包装器中存储指向成员的指针,和/或如果您想将该结构重用于其他类型,则将其设为模板.)

如果你有一个带有 lambdas 的 C++11 编译器,你可以避免所有的样板:

QtConcurrent::map(list, [this] (QString const& data) { add(data); });

注意:我不确定如何QtConcurrent::MemberFunctionWrapper1参与您的示例,我在这里没有看到。因此,对于这种情况,Qt 中可能已经有一个通用包装器,但我不知道。

于 2012-04-28T07:52:56.920 回答