我试图弄清楚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
我尝试了很多不同的方法,无法让它发挥作用,有谁知道原因?提前谢谢!
答案 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);