- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我的问题如下:我需要创建包含 QStateMachine 实例的类。此类应该有插槽,您可以通过这些插槽“请求”状态机转换到另一个状态。如果转换成功,我的类(class)应该发出有关它的信号。我将如何实现?类应该能够根据特定的插槽调用发出特定的信号。这是类的一个小例子:
class MyClass : public QObject
{
Q_OBJECT
public:
explicit MyClass(QObject *parent = 0)
{
mStateMachine = new QStateMachine(this);
QState *s1 = new QState(mStateMachine);
QState *s2 = new QState(mStateMachine);
QState *s3 = new QState(mStateMachine);
s1->addTransition(); // Transition to s2
s2->addTransition(); // Transition to s3
s3->addTransition(); // Transition to s1
mStateMachine->setInitialState(s1);
mStateMachine->start();
}
signals:
toS1();
toS2();
toS3();
public slots:
slotToS1()
{
/* post event to state machine about
transition to state s1,
if transition was successful,
then emit toS1() signal. */
};
slotToS2(){ /* Similar to slotToS1 */};
slotToS3(){ /* Similar to slotToS1 */};
private:
QStateMachine *mStateMachine;
}
非常感谢您的帮助!
更新:
插槽代表不同种类的转换,因此外部类(将使用 MyClass
)可以“请求”一些转换。因此,插槽向状态机发送事件或信号,它查看事件或信号并(如果处于正确状态)进行此转换。我想用某些信号通知外部类,在插槽(转换)成功之前询问。
最佳答案
要在插槽调用上进行转换,您需要以某种方式将插槽绑定(bind)到 QAbstractTransition
。有两种方法:
使用 QEventTransition
并发送相关事件来触发它。
使用 QSignalTransition
并使用内部信号触发它。
要在状态转换时发出信号,您可以连接 QAbstractTransition::triggered
或 QState::entered
或 QState::exited
信号到其他信号。请记住,在 Qt 中,连接目标可以是插槽或信号。
因此,使用信号转换:
class MyClass : public QObject
{
Q_OBJECT
QStateMachine machine;
QState s1, s2;
Q_SIGNAL void s_go_s1_s2();
Q_SIGNAL void s_go_s2_s1();
public:
Q_SIGNAL void transitioned_s1_s2();
Q_SIGNAL void transitioned_s2_s1();
Q_SLOT void go_s2_s1() { emit s_go_s2_s1(); }
Q_SLOT void go_s1_s2() { emit s_go_s1_s2(); }
explicit MyClass(QObject *parent = 0) : QObject(parent),
s1(&machine), s2(&machine) {
auto s1_s2 = s1.addTransition(this, SIGNAL(s_go_s1_s2()), &s2);
auto s2_s1 = s2.addTransition(this, SIGNAL(s_go_s2_s1()), &s1);
machine.setInitialState(&s1);
machine.start();
connect(s1_s2, &QAbstractTransition::triggered, this, &MyClass:: transitioned_s1_s2);
connect(s2_s1, &QAbstractTransition::triggered, this, &MyClass:: transitioned_s2_s1);
}
}
使用事件转换有点困难,因为您正在使用的事件必须可由状态机克隆。核心模块的状态机只知道如何克隆 None
和 Timer
事件 - 请参阅其 cloneEvent
实现。
widgets 模块添加了对各种 GUI/Widgets 事件的支持 - 请参阅 cloneEvent
在那里实现。在紧要关头,您可以将此类 GUI 事件用于您自己的目的 - 毕竟,它们被发送到一个不会以特殊方式解释它们的普通 QObject
。
您可以提供自己的 cloneEvent
实现与其他链接。
#include <private/qstatemachine_p.h>
class MyClass : public QObject
{
Q_OBJECT
QStateMachine machine;
QState s1, s2;
QEvent e_s1_s2, e_s2_s1;
QEventTransition s1_s2, s2_s1;
public:
Q_SIGNAL void transitioned_s1_s2();
Q_SIGNAL void transitioned_s2_s1();
Q_SLOT void go_s2_s1() { QCoreApplication::sendEvent(this, &e_s2_s1); }
Q_SLOT void go_s1_s2() { QCoreApplication::sendEvent(this, &e_s1_s2); }
explicit MyClass(QObject *parent = 0) : QObject(parent),
s1(&machine), s2(&machine),
e_s1_s2((QEvent::Type)(QEvent::User + 1)),
e_s2_s1((QEvent::Type)(QEvent::User + 2)),
s1_s2(this, e_s1_s2.type()),
s2_s1(this, e_s2_s1.type()) {
s1_s2.setTargetState(&s2);
s2_s1.setTargetState(&s1);
s1.addTransition(&s1_s2);
s2.addTransition(&s2_s1);
machine.setInitialState(&s1);
machine.start();
connect(&s1_s2, &QAbstractTransition::triggered, this, &MyClass::transitioned_s1_s2);
connect(&s2_s1, &QAbstractTransition::triggered, this, &MyClass::transitioned_s2_s1);
}
}
static const QStateMachinePrivate::Handler * last_handler = 0;
static QEvent * cloneEvent(QEvent * e) {
if (e->type() >= QEvent::User && e->type() < QEvent::User+100) {
return new QEvent(e->type());
return last_handler->cloneEvent(e);
}
const QStateMachinePrivate::Handler our_handler = {
cloneEvent
};
void registerHandler() {
last_handler = QStateMachinePrivate::handler;
QStateMachinePrivate::handler = &our_handler;
}
Q_CONSTRUCTOR_FUNCTION(registerHandler())
void unregisterHandler() {
QStateMachinePrivate::handler = last_handler;
}
Q_DESTRUCTOR_FUNCTION(unregisterHandler())
关于c++ - 如果 QStateMachine 中的转换成功,则从类发出信号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31385590/
我对带有 ParallelState 的 QStateMachine 很着迷。 我有一个包含第一个状态的主状态(顺序), 包含一组并行状态的第二个状态,以及另一个连续的状态。 第一个状态代表创建 N
我对如何测试 QStateMachine 有点困惑。我有一个组织良好的项目,一侧是源代码,另一侧是测试代码。 标题 class Foo { signals: void sigG
我创建了 4 个不同的 QState,我想在每个状态转换中隐藏一个对象。 这是代码: QStateMachine partita; QState *inizio_mano = new QState()
我已经创建了一个 QStateMachine,我必须获取导致状态转换的事件。难道没有任何机会进入我的插槽 EnterStateInit() 导致此调用的信号。这是我的示例代码: CreateState
在另一个问题中,您告诉我使用 QStateMachine。 我是 Qt 的新手,这是我第一次使用对象,所以我犯了很多逻辑错误,所以使用 QStateMachine 是个大问题...... 这是唯一的方
我的程序中有一个 QStateMachine 实例。我在它的构造函数中配置它的状态、转换和初始状态。当然,我在构造函数中启动它。 this->stateA = new StateA(this); th
为了根据转换设置对象属性,我需要两个可以立即进一步转换的中间状态: A initial state Ta Tb different transitions Aa Ab
我想用 QStateMachine 创建一个无限循环,其中我还需要动画。 QColor leastTransparent, mostTransparent = color(); leastTransp
我正在考虑使用 QStateMachine 在游戏的菜单屏幕之间进行转换。但是,我不确定如何在发生状态转换时启动一些代码(例如 show() a QWidget)。我可以使用普通的旧信号很容易地做到这
以下代码由于内存损坏而导致崩溃。我假设这是因为 delete pTestStateMachine 试图删除未在堆中分配的内存。对吗? 如果是这样,是否意味着 QStateMachine::addSta
我正在尝试弄清楚如何将我自己的自定义信号与 QStateMachine 结合使用。我从 here 中的一个简单示例开始。现在我正在尝试创建一个新信号 mysignal 并触发它的转换。但我不知道如何构
我有这个 Projekt,它使用 QStatemachine 来管理 UI,我想在其中添加自定义列表。 UI 应该只由关键事件操作。据我了解,我需要在 qml 端有一个 ListView。 ListV
我的问题如下:我需要创建包含 QStateMachine 实例的类。此类应该有插槽,您可以通过这些插槽“请求”状态机转换到另一个状态。如果转换成功,我的类(class)应该发出有关它的信号。我将如何实
我目前正在移植 small application从 PyGTK 到位于系统托盘中的 PySide,它会定期检查服务器的更新。 当我最初写这篇文章时,我使用了一个自定义状态模型来表示它的行为: ini
我正在构建一个主要采用顺序流程但也有一些替代路径的程序。我认为状态机可能是最简单的实现方式,因为 Qt 提供了这样一个类:QStateMachine (另见 API)。 不过,我的状态好像挺多的(20
我正在尝试在 Qt (C++) 中实现状态机。如何检查 QStateMachine 的当前状态?我在文档中找不到方法。 谢谢 最佳答案 你试过QStateMachine::configuration(
有一个状态机(称为外部)。这台机器有两个状态——第一个和最后一个。第一个状态是自定义实现的。在第一个状态中创建了另一个状态机(称为内部状态机),在本例中它什么也不做。 具有两个状态的外部状态机:
我正在制作一个应用程序的原型(prototype),我想从应用程序的 C++ 端的 QStateMachine 控制 QML UI 转换。为了使事情更简单,我们可以说 QML UI 由多个页面组成,这
所以我试图了解 Qt 的 QStateMachine 的一个问题,我希望有人能帮助解释为什么会这样。我对 QStateMachine 的基本理解非常感兴趣,而不仅仅是修复。 首先考虑具有状态 A、B
我是一名优秀的程序员,十分优秀!