cv::imshow 在不同的线程上不显示 cv::mat 颜色

cv::imshow does not display cv::mat color when on different thread

本文关键字:cv 显示 mat 颜色 imshow 线程      更新时间:2023-10-16

这是我用来生成一个HighGui窗口的类,其中包含不同线程上的一些内容。

class Capture {
private:
  bool running;
  std::thread thread;
  cv::Mat background;
  void loop() {
    while (running) {
      cv::imshow("sth",background);
      cv::waitKey(settings::capture_wait_time);
    }
  }
  public:
   Capture()  :
     running {false},
     thread {},
     background { 800, 800,  CV_8UC3, cv::Scalar{255,0,255}} {
       cv::namedWindow("sth");  
   }
   inline ~Capture() {
     if (running) stop(); // stop and join the thread
     cv::destroyWindow("sth");
   }
   void run() {
     if (!running) {
       running = true;
       thread = std::thread{[this]{loop();}};
     }
   }
   inline void join() { if (thread.joinable()) thread.join(); };
   inline void stop() {
     running = false;
     if (thread.joinable()) thread.join();
   }
};
// main
Capture cap;
cap.run();
// ... 

问题是窗口最终总是黑色的(在这种情况下应该是紫色的)。我显然在这里错过了一些东西....

您似乎无法在另一个线程中创建窗口。此外,您在另一个线程上调用成员函数的方式似乎是错误的。

看看这段代码。它显示一个在不同线程中每秒更改的图像,并在 5 秒后返回。

#include <opencv2/opencv.hpp>
#include <thread>
using namespace std;
using namespace cv;
class Capture {
private:
    bool running;
    std::thread thread;
    cv::Mat background;
    void loop() {
        while (running) {
            cv::imshow("sth", background);
            cv::waitKey(1000);
            Scalar color(rand()&255, rand()&255, rand()&255);
            background.setTo(color);
        }
    }
public:
    Capture() :
        running{ false },
        thread{},
        background{ 800, 800, CV_8UC3, cv::Scalar{ 255, 0, 255 } } {
    }
    inline ~Capture() {
        if (running) stop(); // stop and join the thread
    }
    void run() {
        if (!running) {
            running = true;
            thread = std::thread{ &Capture::loop, this };
        }
    }
    inline void join() { if (thread.joinable()) thread.join(); };
    inline void stop() {
        running = false;
        if (thread.joinable()) {
            thread.join();
        }
    }
};
int main()
{
    Capture cap;
    cap.run();
    std::this_thread::sleep_for(std::chrono::milliseconds(5000));
    cap.stop();
    return 0;
}