本文主要是介绍Qt信号槽调用方式(emit与QMetaObject::invokeMethod),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
最近才看到在一个线程中接收到数据后需要UI线程刷新时,没有在子线程中直接刷新,而是用到了QMetaObject::invokeMethod的方法。
参考文章:Qt带返回值的信号发射方式(使用QMetaObject::invokeMethod)
QMetaObject::invokeMethod(this, "changeState", Q_ARG(int, 0), Q_ARG(int, m_nSingleState), Q_ARG(int, 0));
void myClass::OutputState(int nType, int nParam1, int nParam2)
{switch (nType){case 0:{}break;case 1:{int nCaptureLength = nParam1;........}break;case 2:{int nTime = nParam1;........}default:break;}
}
connect(this, &myClass::changeState, this, &myClass::OutputState);
1. 为什么需要用invokeMethod而不是emit呢?
一般来说,我们使用invokeMethod是在子线程需要调度UI操作的时候。主要有一下原因:
a. 使用 QMetaObject :: invokeMethod 的原因是收件人对象可能在另一个线程中,如果试图直接在另一个线程中的对象上调用一个槽访问或修改非线程安全数据,则会导致损坏或更糟。
b. 我们可以自己定义连接方式。但是需要注意具体问题具体分析。
2. 简单介绍一下invokeMethod
bool QMetaObject::invokeMethod(QObject *obj,
const char *member,
Qt::ConnectionType type,
QGenericReturnArgument ret,
QGenericArgument val0 = QGenericArgument( Q_NULLPTR ),
QGenericArgument val1 = QGenericArgument(),
QGenericArgument val2 = QGenericArgument(),
QGenericArgument val3 = QGenericArgument(),
QGenericArgument val4 = QGenericArgument(),
QGenericArgument val5 = QGenericArgument(),
QGenericArgument val6 = QGenericArgument(),
QGenericArgument val7 = QGenericArgument(),
QGenericArgument val8 = QGenericArgument(),
QGenericArgument val9 = QGenericArgument())
触发obj对象里面的memeber,member是一个信号/槽的名字。如果正确被触发返回true.
调用可以是同步的,也可以是异步的,具体取决于类型:
如果类型为qt::directconnection,则将立即调用该成员。
如果类型为qt::queuedconnection,则在应用程序进入主事件循环后,将发送qEvent并调用成员。
如果类型为qt::blockingqueuedconnection,则将以与qt::queuedconnection相同的方式调用该方法,但当前线程将阻塞,直到传递事件为止。使用此连接类型在同一线程中的对象之间进行通信将导致死锁。
如果类型为qt::autoconnection,则如果obj与调用方位于同一线程中,则会同步调用该成员;否则,它将异步调用该成员。
成员函数调用的返回值放在ret中。如果调用是异步的,则无法计算返回值。最多可以向成员函数传递十个参数(val0、val1、val2、val3、val4、val5、val6、val7、val8和val9)。
QGenericArgument and QGenericReturnArgument are internal helper classes. Because signals and slots can be dynamically invoked, you must enclose the arguments using the Q_ARG() and Q_RETURN_ARG() macros. Q_ARG() takes a type name and a const reference of that type; Q_RETURN_ARG() takes a type name and a non-const reference.
QGenericArgument和QGenericReturnArgument是内部帮助程序类。由于可以动态调用信号和插槽,因此必须使用q_arg()和q_return_arg()宏将参数括起来。q_arg()接受类型名和该类型的常量引用;q_return_arg()接受类型名和非常量引用。
这篇关于Qt信号槽调用方式(emit与QMetaObject::invokeMethod)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!