TemporalBeing
20th March 2009, 22:27
I've got an application that receive a lot of data over the network via a QTcpSocket. If I keep the it all in the main thread, then the processor usage spikes and memory goes through the roof as there is no time left for processing events. I am trying to spin this processing off into a new object derived from QThread, and then use signals/slots to talk with the main thread.
The derived thread primarily does the following:
1) Receives data from the QTcpSocket.
2) Validates the data
3) Prepares the data for display
4) Sends 1 piece of data to the main application.
5) Upon receipt of #4, the main application sends the old data (if any) back to the thread for destruction.
// myObjectPtr is registered with qRegisterMetaType()
// dataInfoObject is registered with qRegisterMetaType()
// it's copy constructor initializes its QObject parent to NULL.
class myThread : public QThread
{
Q_OBJECT
Q_SIGNALS:
void sendData(const myObjectPtr data);
public Q_SLOTS:
void setInfo(const dataInfoObject& _objectInfo);
void setServer(const QString& _hostAddress, quint16 _hostPort);
void getDataBack(const myObjectPtr data);
protected Q_SLOTS:
void dataAvailable();
protected:
// TCP Data STream
QTcpSocket* myTcpConnection;
// QObject derived class for information about the thread
dataInfoObject* myInfo;
void myThreadInit();
public:
myThread();
~myThread();
void run();
};
...
myThread::myThread()
{
myInfo = NULL;
myTcpConnection = NULL;
}
void myThread::run()
{
myThreadInit();
exec();
}
void myThread::myThreadInit()
{
myInfo = new dataInfoObject(this);
// connect internal stuff
}
void myThread::setServer(const QString& _hostAddress, quint16 _hostPort)
{
if (myTcpConnection == NULL)
{
myTcpConnection = new QTcpSocket(this);
if (myTcpConnection != NULL)
{
// connect stuff up here - like readyRead(), connected(), etc.
}
}
if (myTcpConnection != NULL)
{
myTcpConnection->connectToHost(_hostAddress,_port);
}
}
...
The derived thread's run() simply calls its exec(), which to my understanding is needed to use the thread's event queue and signals/slots.
The main thread creates a widget (which manages the display of the data), which creates the thread as part of its construction. After the thread is created, it connects several signals/slots for the two parts to communicate and then calls the thread's start.
class myWidget : public QWidget
{
Q_OBJECT
...
private Q_SLOTS:
void getData(const myObjectPtr data);
Q_SIGNALS:
void sendDataBack(const myObjectPtr data);
void setThreadInfo(const dataInfoObject& _objectInfo);
private:
myThread workerThread;
void createThread();
protected:
...
...
public:
myWidget(QWidget* parent=NULL);
};
...
myWidget::myWidget(QWidget* parent) : QWidget(parent)
{
workerThread = NULL;
...
createThread();
}
...
void myWidget::createThread()
{
if (workerThread == NULL)
{
workerThread = new myThread(this);
}
if (workerThread != NULL)
{
// connect stuff up - connection type not specified
connect(this,SIGNAL(setThreadInfo(const dataInfoObject&)),workerThread,SLOT(setInfo(const dataInfoObject&)));
...
workerThread->start();
Q_EMIT setThreadInfo(threadInfo);
}
}
...
I just implemented the QThread in the last couple days; however, I discovered today that all the signal/slots are being handled in the main thread context, not the child thread's context. Thus the main thread is still chewing up all the processing time.
All the data getting passed over the signals/slots is either native to Qt, native C++ (e.g. bool, unsigned int, etc.), or it's been registered with the MetaObject system (via Q_DECLARE_METATYPE() and qRegisterMetaType()) so that it can be copied by Qt's event system.
I don't have any locks (yet) as I have not yet seen a need for them - the data is either in one context or the other, not both. I have also tried using the moveToThread() on the data object before emitting the signal with the object.
...
dataObject->(QApplication::instance()->thread());
Q_EMIT sendData(dataObject);
...
One thing I have found is that when I initialize the data in the thread in myThreadInit() - I will get the error:
QObject: Cannot create children for a parent that is in a different thread.
if I provide a parent to the object even though it is creating it from within the thread's context.
// Generates: QObject: Cannot create children for a parent that is in a different thread.
myInfo = new dataInfoObject(this);
// doesn't generate: QObject: Cannot create children for a parent that is in a different thread.
myInfo = new dataInfoObject();
What am I doing wrong? What am I misunderstanding about QThreads?
The derived thread primarily does the following:
1) Receives data from the QTcpSocket.
2) Validates the data
3) Prepares the data for display
4) Sends 1 piece of data to the main application.
5) Upon receipt of #4, the main application sends the old data (if any) back to the thread for destruction.
// myObjectPtr is registered with qRegisterMetaType()
// dataInfoObject is registered with qRegisterMetaType()
// it's copy constructor initializes its QObject parent to NULL.
class myThread : public QThread
{
Q_OBJECT
Q_SIGNALS:
void sendData(const myObjectPtr data);
public Q_SLOTS:
void setInfo(const dataInfoObject& _objectInfo);
void setServer(const QString& _hostAddress, quint16 _hostPort);
void getDataBack(const myObjectPtr data);
protected Q_SLOTS:
void dataAvailable();
protected:
// TCP Data STream
QTcpSocket* myTcpConnection;
// QObject derived class for information about the thread
dataInfoObject* myInfo;
void myThreadInit();
public:
myThread();
~myThread();
void run();
};
...
myThread::myThread()
{
myInfo = NULL;
myTcpConnection = NULL;
}
void myThread::run()
{
myThreadInit();
exec();
}
void myThread::myThreadInit()
{
myInfo = new dataInfoObject(this);
// connect internal stuff
}
void myThread::setServer(const QString& _hostAddress, quint16 _hostPort)
{
if (myTcpConnection == NULL)
{
myTcpConnection = new QTcpSocket(this);
if (myTcpConnection != NULL)
{
// connect stuff up here - like readyRead(), connected(), etc.
}
}
if (myTcpConnection != NULL)
{
myTcpConnection->connectToHost(_hostAddress,_port);
}
}
...
The derived thread's run() simply calls its exec(), which to my understanding is needed to use the thread's event queue and signals/slots.
The main thread creates a widget (which manages the display of the data), which creates the thread as part of its construction. After the thread is created, it connects several signals/slots for the two parts to communicate and then calls the thread's start.
class myWidget : public QWidget
{
Q_OBJECT
...
private Q_SLOTS:
void getData(const myObjectPtr data);
Q_SIGNALS:
void sendDataBack(const myObjectPtr data);
void setThreadInfo(const dataInfoObject& _objectInfo);
private:
myThread workerThread;
void createThread();
protected:
...
...
public:
myWidget(QWidget* parent=NULL);
};
...
myWidget::myWidget(QWidget* parent) : QWidget(parent)
{
workerThread = NULL;
...
createThread();
}
...
void myWidget::createThread()
{
if (workerThread == NULL)
{
workerThread = new myThread(this);
}
if (workerThread != NULL)
{
// connect stuff up - connection type not specified
connect(this,SIGNAL(setThreadInfo(const dataInfoObject&)),workerThread,SLOT(setInfo(const dataInfoObject&)));
...
workerThread->start();
Q_EMIT setThreadInfo(threadInfo);
}
}
...
I just implemented the QThread in the last couple days; however, I discovered today that all the signal/slots are being handled in the main thread context, not the child thread's context. Thus the main thread is still chewing up all the processing time.
All the data getting passed over the signals/slots is either native to Qt, native C++ (e.g. bool, unsigned int, etc.), or it's been registered with the MetaObject system (via Q_DECLARE_METATYPE() and qRegisterMetaType()) so that it can be copied by Qt's event system.
I don't have any locks (yet) as I have not yet seen a need for them - the data is either in one context or the other, not both. I have also tried using the moveToThread() on the data object before emitting the signal with the object.
...
dataObject->(QApplication::instance()->thread());
Q_EMIT sendData(dataObject);
...
One thing I have found is that when I initialize the data in the thread in myThreadInit() - I will get the error:
QObject: Cannot create children for a parent that is in a different thread.
if I provide a parent to the object even though it is creating it from within the thread's context.
// Generates: QObject: Cannot create children for a parent that is in a different thread.
myInfo = new dataInfoObject(this);
// doesn't generate: QObject: Cannot create children for a parent that is in a different thread.
myInfo = new dataInfoObject();
What am I doing wrong? What am I misunderstanding about QThreads?