Qt invokeMethod调用具有输出参数的函数

时间:2018-02-24 03:45:42

标签: c++ qt qt5 qmetaobject

我试图弄清楚QMetaObject :: invokeMethod的用法。我有一个函数有一个参数(非const QString),我希望它是输出,函数没有返回值,调用invokeMethod总是失败,而另一个函数有返回值,没有参数可以被调用成功。以下是代码:

myclass.h

#include <QDebug>

class MyClass: public QObject
{
    Q_OBJECT
public:
    MyClass() {}
    ~MyClass() {}
public slots:
    QString func();
    void func2(QString& res);
};

myclass.cpp

#include "myclass.h"

QString MyClass::func()
{
    QString res = "func succeeded";
    qDebug() << res;
    return res;
}

void MyClass::func2(QString& res)
{
    res = "func2 succeeded";
    qDebug() << res;
    return;
}

的main.cpp

#include <QCoreApplication>
#include "myclass.h"

int main(int argc, char *argv[])
{
    QString msg;
    MyClass myobj;

    QCoreApplication a(argc, argv);

    bool val = QMetaObject::invokeMethod(&myobj, "func", Q_RETURN_ARG(QString, msg));
    qDebug() << "func returns" << val;

    val = QMetaObject::invokeMethod(&myobj, "func2", Q_RETURN_ARG(QString, msg));
    qDebug() << "func2 returns" << val;

    int ret = a.exec();
    return ret;
}

结果如下:

$ ./test
"func succeeded"
func returns true
QMetaObject::invokeMethod: No such method MyClass::func2()
Candidates are:
    func2(QString&)
func2 returns false

我尝试了很多不同的方法,无法让它发挥作用,有谁知道原因?提前谢谢!

2 个答案:

答案 0 :(得分:1)

使用Q_RETURN_ARG时获取方法返回的值,但是如果要传递一些参数,则必须使用Q_ARG

#include <QCoreApplication>
#include "myclass.h"

int main(int argc, char *argv[])
{
    QString msg;
    MyClass myobj;

    QCoreApplication a(argc, argv);

    bool val = QMetaObject::invokeMethod(&myobj, "func", Q_RETURN_ARG(QString, msg));
    qDebug() << "func returns" << val;

    val = QMetaObject::invokeMethod(&myobj, "func2", Q_ARG(QString &, msg));
    qDebug() << "func2 returns" << val;

    int ret = a.exec();
    return ret;
}

输出:

"func succeeded"
func returns true
"func2 succeeded"
func2 returns true

答案 1 :(得分:1)

错误消息说明了一切。该类没有签名MyClass:func2()的方法。

您已声明该函数具有签名:void func2(QString&),这与QString& func2()不同。您使用函数的参数作为返回值的事实与Qt信号和插槽代码命名方法或查找它们的方式无关,事实上,与此无关呼叫签名。呼叫签名就是它的任何东西,Qt无法理解您打算如何使用这些参数。

最简单的解决方法是更改​​调用方法的方式。你需要

QMetaObject::invokeMethod(&myobj, "func2", Q_ARG(QString, msg));

或者更好的是,使用Qt 5.0+类型的安全信号和插槽。

QMetaObject::invokeMethod(&myobj, &MyClass::func2, &msg);