[Solved] QThread blocking GUI
-
Move everything you do in the constructor of the Worker into the MainWindow::createWorker() function. You are not allowed to do anything GUI related in another thread then the main thread.
Did you already see "this example":http://qt-project.org/wiki/QThreads_general_usage for a worker object approach with QThread ? -
[quote author="KA51O" date="1357718103"]Move everything you do in the constructor of the Worker into the MainWindow::createWorker() function. You are not allowed to do anything GUI related in another thread then the main thread.
Did you already see "this example":http://qt-project.org/wiki/QThreads_general_usage for a worker object approach with QThread ?[/quote]I already read this example from the original blog.
I also moved everything gui-related from my Worker to my MainWindow.
Nothing changed, I really don't understand : now the Gui object is owned by the MainWindow and everything is handled by signals...@mainwindow.cpp
void MainWindow::createWorker()
{worker = new Worker();
worker_gui = new Gui();
thread = new QThread();worker->moveToThread(thread);
ui->worker_widget = worker_gui;
connect(worker_gui, SIGNAL(start()), worker, SLOT(read_data()));
connect(worker_gui, SIGNAL(stop()), worker, SLOT(pause_processing()));
connect(worker, SIGNAL(emitData(QStringList)), worker_gui, SLOT(setRow(QStringList)));thread->start();
}@
-
process_data() recieves the data from the serial bus using a Communication class and adapts it to the desired format. It finally sends the data to the gui using emit_data(QStringList).
What surprises me is that the old design worked, except for the segfaults (probably coming from the non-thread-safe implementation).
The worker code wouldn't be relevant as there is nothing in the constructor now. The main concern would be the while(running) infinite loop in read_data().
-
Infinite loops in an event-driven framework are not really fitting. They block the event queue! Maybe in the old version this was no problem because the worker (or subclassed thread or whatever) didn't need to receive any signals or didn't rely on the eventloop.
So the last thing I can recommend would be to try and refactor the code so you don't need the inifinite loop (try solving it with signals and slots). -
Ok thank you very much.
My gui now reacts, I used the QCoreApplication::processEvents() that I found "here":http://qt-project.org/wiki/Threads_Events_QObjects. So now my signals are correctly sent.
But that does not seem to be the correct way of handling things and, as you said I still don't understand why it kept blocking the main thread. -
Do you call anything that is living in the context of the main thread directly (not using signal slot) from out of the process_data() function in your worker thread ?
Be aware that calling QCoreApplication::processEvents() in an infinite loop will raise the processor workload to 100 %.
EDIT: It almost seems as if the infinite loop in your worker is running in the context of the main thread (although you moved it to the worker thread).
-
I just finished refactoring the read and pause features (now I need to refactor all the others so they are complying to this model). Everything is now handled by signals.
I remplaced the infinite loop by a QTimer so now I can stop() and start() in an easy and most convenient way.