为什么我不能从基类插槽Qt调用虚函数

Why can not I call virtual function from base class slot Qt

本文关键字:Qt 调用 函数 插槽 基类 不能 为什么      更新时间:2023-10-16

有人可以向我解释为什么在基类插槽中没有调用被覆盖的方法,而是我有一个基本版本的方法:

class ThreadsDispatcher : public QObject
{
   Q_OBJECT
   public:
      explicit ThreadsDispatcher(QObject *parent = 0);
      virtual ~ThreadsDispatcher();
      virtual void OnThreadFinished(IThreadable *pWorker);
   public slots:
      void slotThreadFinished(IThreadable *pWorker);
};
void ThreadsDispatcher::slotThreadFinished(IThreadable *pWorker)
{
   OnThreadFinished(pWorker);
}
void ThreadsDispatcher::OnThreadFinished(IThreadable *pWorker)
{
   qDebug << "Base method, class" << this->metaObject()->className();
}

子类:

class CommandsQueueDispatcher : public ThreadsDispatcher
{
    Q_OBJECT
    public:
       explicit CommandsQueueDispatcher(CommandFactory* baseFactory, QObject *parent = 0);
       ~CommandsQueueDispatcher();
       void OnThreadFinished(IThreadable *pWorker);
};
 void CommandsQueueDispatcher::OnThreadFinished(IThreadable *pWorker)
 {
    qDebug << "Subclass method, class" << this->metaObject()->className();
 }

调用后 线程完成 在插槽中,我得到:

Base method, class ThreadsDispatcher

如果我从另一个方法调用方法 OnThreadDone,我会得到正常:

Subclass method, class CommandsQueueDispatcher

我尝试在基类和子类中连接,但没有变化:

connect(pThreadWorker, SIGNAL(sigFinished(IThreadable*)), this, SLOT(slotThreadFinished(IThreadable*)));

但是如果我从另一个类连接,即既不是子类,也不是基类:

    connect(pThreadWorker, SIGNAL(sigFinished(IThreadable*)), pWorker, SLOT(slotThreadFinished(IThreadable*)));

我需要用变量ptr替换this的地方,我得到一个正常的结果。

我连接的功能:

bool ThreadsDispatcher::AddThread(IThreadable* pThreadWorker)
{
   connect(pThreadWorker, SIGNAL(sigFinished(IThreadable*)), this, SLOT(slotThreadFinished(IThreadable*)));
}

我不直接实例化ThreadsDispatcher。我创建了 CommandQueueDispatcher 的非静态对象。

故障排除建议(太长,无法添加注释):

尝试将插槽更改为如下所示:

void ThreadsDispatcher::OnThreadFinished(IThreadable *pWorker)
{
   qDebug << "Base method, class" << this->metaObject()->className();
}

派生类也是如此。查看输出内容。

如果输出是"正确的",因为调用的虚拟方法与类名一致,那么我怀疑你确实以某种方式有一个基类对象,而不是派生的类对象。

如果存在输出不匹配,基类虚拟方法执行打印但报告派生的类名,那么我会寻找任何有趣的编译器标志,并尝试使用新项目创建一个 SSCCE,然后也许在这里再次询问和/或向 Qt 提交错误报告。

在Qt文档中开始阅读此内容的链接。


故障排除的第二个步骤:通过将此方法在基类中排除来将基类更改为抽象:

  virtual void OnThreadFinished(IThreadable *pWorker) = 0;

。然后删除方法定义。现在编译器应该告诉你在哪里尝试创建基类的实例。


还有一个建议:Qt构建基础知识,例如在此答案中列出。特别是,请确保 QObject 子类是在 .h 文件中定义的,这些文件列在 .pro 文件 HEADERS 列表中。