我很擅长使用QThread
。我正在使用QThread
从Axis Ip相机中抓取图像。在下面的代码片段中,我将相机类移动到一个新线程:
QThread camThread;
camera = new AxisNetworkCamera();
camera->moveToThread(&camThread);
camThread.start();
connect(camera, SIGNAL(imageUpdate(QImage)), this, SLOT(upDateImage(QImage)));
connect(camera, SIGNAL(cameraDisconnected()), this, SLOT(cameraDisconnected()));
connect(&camThread, &QThread::finished, &camThread, &QThread::deleteLater);
connect(camera, &AxisNetworkCamera::destroyed, &camThread, &QThread::quit);
我正在调用启动相机的功能:
QMetaObject::invokeMethod(camera, "deviceStartup", Qt::QueuedConnection, Q_ARG(QString, QString::fromStdString(streamUrl)));
应用程序运行良好,关闭时也关闭,但我担心的是一些警告消息。
首先是我启动相机时:
Type conversion already registered from type QPair<QByteArray,QByteArray> to type QtMetaTypePrivate::QPairVariantInterfaceImpl
第二个是我关闭申请时:
QThreadStorage: Thread 0x7fffb8004da0 exited after QThreadStorage 7 destroyed
我应该担心这些消息吗?他们,特别是第二个,是否意味着任何记忆管理错误?
由于
答案 0 :(得分:2)
您发布的代码毫无意义。 QThread
未动态分配,因此您无法delete
:deleteLater
调用将崩溃。可能它永远不会被执行,因为你没有显示任何会阻止线程的代码。在相机被摧毁后,破坏线程也毫无意义。
安全地执行操作的最简单方法是在类中按值保存摄像机和线程,并按正确的顺序声明它们,以便在摄像机之前销毁线程。此时,相机变得无线,可以安全地在任何线程中销毁。
在远程线程中调用方法的方法比使用invokeMethod
更好:
class Thread : public QThread {
using QThread::run; // final
public:
Thread(QObject*parent = 0): QThread(parent) {}
~Thread() { quit(); wait(); }
};
// See http://stackoverflow.com/a/21653558/1329652 for details about invoke.
template <typename F> void invoke(QObject * obj, F && fun) {
if (obj->thread == QThread::currentThread())
return fun();
QObject src;
QObject::connect(&src, &QObject::destroyed, obj, std::forward<F>(fun));
}
class MyObject : public QObject {
Q_OBJECT
AxisNetworkCamera camera;
Thread camThread { this };
Q_SLOT void updateImage(const QImage &);
Q_SLOT void cameraDisconnected();
public:
MyObject(QObject * parent = 0) : QObject(parent)
{
connect(&camera, &AxisNetworkCamera::imageUpdate, this, &MyObject::updateImage);
connect(&camera, &AxisNetworkCamera::cameraDisconnected, this, &MyObject::cameraDisconnected);
camera.moveToThread(&camThread);
camThread.start();
}
void startCamera(const QString & streamUrl) {
invoke(&camera, [=]{ camera.deviceStartup(streamUrl); });
}
};