在信号中使用enum
类型时遇到一些问题。基本上我有两个类,一个状态机和一个处理状态机的线程。当状态改变时,我想发送一个具有新状态的信号。我还想用enum
代表州。在我的完整代码中,状态机是在一个单独的共享库中实现的,但下面的代码给出了完全相同的错误。
当我运行代码时,我得到以下行为:
kotte@EMO-Ubuntu:sigenum $ ./sigenum
Object::connect: No such slot MyThread::onNewState(state)
Test signal
Test signal
...
我的示例代码中有四个文件:statemachine.h
,statemachine.cpp
,main.h
和main.cpp
。 main函数只是启动线程,然后线程创建StateMachine
的实例并处理来自StateMachine
的信号。我对Qt很新,所以当我意识到你必须用Q_ENUMS
括起枚举并将其注册到类型系统时,我有点困惑。所以我完全有可能犯了一些菜鸟错误
下面的代码有点长,但我希望它与我的真实代码尽可能相似。
statemachine.h
看起来像:
// statemachine.h
#ifndef _STATEMACHINE_H
#define _STATEMACHINE_H
#include <QtCore>
class StateMachine : public QObject
{
Q_OBJECT
Q_ENUMS(state)
public:
enum state {S0, S1, S2};
void setState(state newState);
signals:
void stateChanged(state newState);
void testSignal(void);
};
Q_DECLARE_METATYPE(StateMachine::state);
#endif
它实现为:
// statemachine.cpp
#include <QtCore>
#include "statemachine.h"
void StateMachine::setState(state newState)
{
emit stateChanged(newState);
emit testSignal();
}
线程定义为
// main.h
#ifndef _MAIN_H
#define _MAIN_H
#include <QtCore>
#include "statemachine.h"
class MyThread : public QThread
{
Q_OBJECT
private:
void run(void);
private slots:
void onNewState(StateMachine::state);
void onTestSignal(void);
private:
StateMachine *myStateMachine;
};
#endif
它实现如下:
// main.cpp
#include <QtCore>
#include <QApplication>
#include "statemachine.h"
#include "main.h"
void MyThread::run()
{
myStateMachine = new StateMachine();
qRegisterMetaType<StateMachine::state>("state");
// This does not work
connect(myStateMachine, SIGNAL(stateChanged(state)),
this, SLOT(onNewState(state)));
// But this does...
connect(myStateMachine, SIGNAL(testSignal()),
this, SLOT(onTestSignal()));
forever {
// ...
myStateMachine->setState(StateMachine::S0);
}
}
void MyThread::onTestSignal()
{
qDebug() << "Test signal";
}
void MyThread::onNewState(StateMachine::state newState)
{
qDebug() << "New state is:" << newState;
}
答案 0 :(得分:16)
通过在任何地方使用完全限定名称,我开始工作
如果我将stateChanged()
的声明更改为
signals:
void stateChanged(StateMachine::state newState);
并使用
注册类型qRegisterMetaType<StateMachine::state>("StateMachine::state");
并在connect
声明中使用此名称
connect(myStateMachine, SIGNAL(stateChanged(StateMachine::state)),
this, SLOT(onNewState(StateMachine::state)));
如果没有来自drescherjm的输入,就不会解决这个问题,谢谢: - )
答案 1 :(得分:4)
我认为以下状态未在MyThread类中定义。
使用以下
connect(myStateMachine, SIGNAL(stateChanged(StateMachine::state)), this, SLOT(onNewState(StateMachine::state)));
编辑:
也许这会起作用
connect(myStateMachine, SIGNAL(stateChanged(state)), this, SLOT(onNewState(StateMachine::state)));
答案 2 :(得分:0)
您应该摆脱SIGNAL
和SLOT
,因为Qt可以在编译时检测到不匹配。您还可以避免使用Q_ENUM而不是Q_DECLARE_METATYPE
来使用qRegisterMetaType()
和Q_ENUMS
-这是Qt 5.5中引入的,最后enum class
是一种强类型enum
的版本:
// statemachine.h
#ifndef _STATEMACHINE_H
#define _STATEMACHINE_H
#include <QtCore>
class StateMachine : public QObject
{
Q_OBJECT
Q_ENUM(state)
public:
enum class state {S0, S1, S2};
void setState(state newState);
signals:
void stateChanged(state newState);
void testSignal(void);
};
#endif
// main.cpp
#include <QtCore>
#include <QApplication>
#include "statemachine.h"
#include "main.h"
void MyThread::run()
{
myStateMachine = new StateMachine();
connect(myStateMachine, &StateMachine::stateChanged, this, &MyThread::NewState);
connect(myStateMachine, &StateMachine::testSignal, this, &MyThread::onTestSignal);
forever {
// ...
myStateMachine->setState(StateMachine::S0);
}
}