我对使用QThread
很陌生。我正在使用QThread
从安讯士 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
我应该担心这些消息吗?它们,特别是第二个1,是否意味着任何内存管理不善?
谢谢
你发布的代码没有意义。 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); });
}
};