Tom*_*ica 5 c++ qt signals-slots queued-connection
我有这样的课:
#include <QObject>
namespace taservices
{
class ProcessHandle : public QObject
{
Q_OBJECT
public:
ProcessHandle(const void* const processContextPointer, const QString& process_id = "", QObject *parent = 0);
ProcessHandle();
signals:
void progress(const ProcessHandle* const self, const int value);
private:
static void registerAsMetaType();
}
Run Code Online (Sandbox Code Playgroud)
我有一个信号:
void progress(const ProcessHandle* const self, const int value);
Run Code Online (Sandbox Code Playgroud)
我想通过QueuedConnedtion连接它。我不断收到此消息:
QObject::connect: Cannot queue arguments of type 'ProcessHandle*const'
(Make sure 'ProcessHandle*const' is registered using qRegisterMetaType().)
Run Code Online (Sandbox Code Playgroud)
声明后,我像这样注册我的课程:
Q_DECLARE_METATYPE(taservices::ProcessHandle*);
Run Code Online (Sandbox Code Playgroud)
我还添加了从构造函数调用的此静态方法:
void ProcessHandle::registerAsMetaType()
{
static bool called = false;
if(!called) {
called = true;
qRegisterMetaType<ProcessHandle*>("taservices::ProcessHandle*");
}
}
Run Code Online (Sandbox Code Playgroud)
我也尝试注册const指针:
qRegisterMetaType<ProcessHandle*const>("taservices::ProcessHandle*const");
Run Code Online (Sandbox Code Playgroud)
它导致以下错误:
error C2440: 'return' : cannot convert from 'taservices::ProcessHandle *const *' to 'void *'
Run Code Online (Sandbox Code Playgroud)
那么,如何使我的班级使用排队连接呢?
事实证明这就是您所需要的:
qRegisterMetaType<ProcessHandle*>("ProcessHandle*");
qRegisterMetaType<ProcessHandle*>("ProcessHandle*const");
Run Code Online (Sandbox Code Playgroud)
出于排队参数的目的,const指针等于普通指针,因为它是被复制而不是改变的。
带有const值参数的信号毫无意义。常量的唯一一点是防止实现行为不当和修改值,这意味着不应以某种方式修改该值(为什么不呢?这是您泄漏到接口中的实现细节!)。信号的代码是由 moc 生成的,如果此类代码行为不当,那么您已经遇到了更大的问题。
您的信号应具有以下声明:
Q_SIGNAL void progress(const ProcessHandle* self, int value);
Run Code Online (Sandbox Code Playgroud)
该槽可以自由地具有 const 参数。就 Qt 而言,最内部的常量不是签名的一部分 - 它被有效地删除了。
您不需要注册类型。当您让 connect 使用新语法访问类型时,它会自动完成connect。
例子:
// https://github.com/KubaO/stackoverflown/tree/master/questions/const-slot-arg-42163294
#include <QtCore>
struct ProcessHandle {};
struct Object : QObject {
int counter = 0;
Q_SIGNAL void newValue(const ProcessHandle*, int val);
Q_SLOT void useValue(const ProcessHandle* const ph, const int val) {
qDebug() << ph << val;
Q_ASSERT(ph == nullptr && val == 42);
++counter;
}
Q_OBJECT
};
int main(int argc, char ** argv) {
QCoreApplication app{argc, argv};
Object obj;
QObject::connect(&obj, &Object::newValue, &obj, &Object::useValue, Qt::QueuedConnection);
QObject::connect(&obj, &Object::newValue, &app, &QCoreApplication::quit, Qt::QueuedConnection);
emit obj.newValue(nullptr, 42);
app.exec();
Q_ASSERT(obj.counter == 1);
}
#include "main.moc"
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
2509 次 |
| 最近记录: |