使用QThread时输入转换和QThreadStorage警告?

时间:2016-05-27 09:12:51

标签: c++ qt qthread

我很擅长使用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

我应该担心这些消息吗?他们,特别是第二个,是否意味着任何记忆管理错误?

由于

1 个答案:

答案 0 :(得分:2)

您发布的代码毫无意义。 QThread未动态分配,因此您无法deletedeleteLater调用将崩溃。可能它永远不会被执行,因为你没有显示任何会阻止线程的代码。在相机被摧毁后,破坏线程也毫无意义。

安全地执行操作的最简单方法是在类中按值保存摄像机和线程,并按正确的顺序声明它们,以便在摄像机之前销毁线程。此时,相机变得无线,可以安全地在任何线程中销毁。

在远程线程中调用方法的方法比使用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); });
  }
};