使用 QThread 时的类型转换和 QThreadStorage 警告



我对使用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); });
  }
};

最新更新