是否可以在Qt(5.x)中实现函数中断?
例如,如果我有一个按钮并希望在单击此按钮时在线程(正在运行无限循环)上执行某些内容,我可以这样说:
in thread...
forever
{
if(button_is_pressed_flag)
{
do something...
}
}
有没有更好的方法?
无限循环应该是一个事件循环,然后它可以自动处理跨线程插槽调用,而无需担心细节。
在事件循环上"连续"运行代码的习惯用法是零持续时间计时器。
假设您从如下所示的代码开始:
class MyThread : public QThread {
bool button_is_clicked_flag = false;
void run() override {
forever{
if (button_is_clicked_flag) {
onButtonClick();
button_is_clicked_flag = false;
}
doWork();
}
}
void onButtonClick();
void doWork();
public:
using QThread::QThread;
void setButtonClickedFlag();
}
int main(int argc, char **argv) {
...
MyThread t;
t.start();
...
}
要求doWork()
不要花太长时间 - 也不要太短。如果在现代硬件上花费~5ms,那么对于通用应用程序来说,这将是开销和延迟之间的正确权衡。如果需要在工作线程中降低延迟反应,则doWork()
必须执行更少的工作。对于doWork()
来说,花费的时间远少于 1 毫秒可能没有多大意义。
每当doWork()
没有任何可做的时候,例如,如果它完成了它应该执行的计算,它应该停止保持它活着的计时器。
应将其转换为如下所示:
class MyWorker : public QObject {
Q_OBJECT
QBasicTimer m_timer;
void doWork();
void timerEvent(QTimerEvent *event) {
if (event->timerId() == m_timer.timerId())
doWork();
}
public:
explicit MyWorker(QObject *parent = nullptr) : QObject(parent) {
m_timer.start(0, this);
}
Q_SLOT void onButtonClick() {
// Ensure we're invoked correctly
Q_ASSERT(QThread::currentThread() == thread());
...
}
}
class Window : public QWidget {
Ui::Window ui;
public:
Q_SIGNAL void buttonClicked();
explicit Window(QWidget *parent = nullptr) : QWidget(parent) {
ui.setupUi(this);
connect(ui.button, &QPushButton::clicked, this, &Window::buttonClicked);
}
};
class SafeThread : public QThread {
Q_OBJECT
using QThread::run; // final method
public:
~SafeThread() { quit(); wait(); } // we're safe to destroy - always
};
int main(int argc, char **argv) {
...
MyWorker worker;
SafeThread thread;
Window window;
// onButtonClick will be executed in worker->thread()
connect(&window, &Window::buttonClicked, &worker, &MyWorker::onButtonClick);
worker.moveToThread(&thread);
thread.start();
window.show();
return app.exec();
}
在QThread::run
中运行的事件循环将通过计时器事件处理程序连续调用doWork
。但是每当需要对位于该线程中的对象进行跨线程时隙调用时,事件循环会将表示槽调用的内部QMetaCallEvent
传递给QObject::event
,然后执行调用。
因此,当您在onButtonClick
中设置断点时,调用堆栈上将有QObject::event
。
您可以启动一个线程,然后立即等待 std::condition_variable,然后在单击按钮(在主线程上调用的事件)时,通知条件变量,线程将唤醒。
但是,这有点奇怪。你想做什么?单击按钮时调用异步任务?在这种情况下,也许最好从带有 std::p ackaged_task 或 std::async 的按钮单击事件开始一个。