如何将灰度/二进制 Mat 转换为 QImage?

Posted

技术标签:

【中文标题】如何将灰度/二进制 Mat 转换为 QImage?【英文标题】:How to convert Grayscale/binary Mat to QImage? 【发布时间】:2016-11-22 12:42:42 【问题描述】:

我已经开始学习 Qt 并且正在尝试制作一个简单的视频播放器来加载视频并播放它。它工作得很好。现在为其添加了阈值功能。阈值将从 spinBox 中获取。 代码以这样一种方式编写,即阈值操作将使用 spinBox 中的值完成,但值 0(显示正常视频的位置)除外。 所以这是我的功能:

void Player::run()


while(!stop )


    if(!capture.read(frame))
        stop = true;
    // convert RGB to gray
    if(frame.channels() == 3)
    
        if(thresh == 0)
         
            cvtColor(frame, RGBframe, CV_BGR2RGB);
            img = QImage((const unsigned char*)(RGBframe.data),
                          RGBframe.cols,RGBframe.rows,QImage::Format_RGB888);

         
        else
        
            Mat temp;
            cvtColor(frame, temp, CV_BGR2GRAY);
            threshold(temp, binary, thresh, 255, 0);
            img = QImage((const unsigned char*)(binary.data),
                              binary.cols, binary.rows, QImage::Format_Indexed8);




            bool save = img.save("/home/user/binary.png");

             cout<<"threshold value = "<<thresh<<endl;

            //imshow("Binary", binary);
        
     
     else
     
        if(thresh == 0) // original Image
        
            img = QImage((const unsigned char*)(frame.data),
                             frame.cols,frame.rows,QImage::Format_Indexed8);

        
        else // convert to Binary Image
        

            threshold(frame, binary, thresh, 255, 0);

            img = QImage((const unsigned char*)(binary.data),
                              binary.cols, binary.rows, QImage::Format_Indexed8);
        




     

     emit processedImage(img);
     this->msleep(delay);

  

对于 spinBox 值等于 0,它运行良好,但是当 spinBox 值增加时,我只得到黑屏。我尝试了imshow(cv:: Mat binary),它显示了正确的二进制图像,但是当我尝试保存QImage img 时,它是一些随机的黑白像素(尽管与原始帧大小相同)。

【问题讨论】:

您似乎缺少一个颜色表。在 while 循环之前添加:QVector&lt;QRgb&gt; sColorTable(256); for (int i = 0; i &lt; 256; ++i)sColorTable[i] = qRgb(i, i, i);,在从二进制文件创建 QImage 之后,添加 img.setColorTable(sColorTable); 彩色图像输入案例还是灰度图像输入案例?在 Qt 中,灰度不是 indexed8 类型,我不得不使用颜色表。 【参考方案1】:

您似乎缺少索引图像的颜色表。您需要添加一个颜色表(在while循环之前):

 QVector<QRgb> sColorTable(256); 
 for (int i = 0; i < 256; ++i) sColorTable[i] = qRgb(i, i, i); 

从二进制Mat 创建QImage 后,您需要添加

 img.setColorTable(sColorTable);

或者,正如@KubaOber 所指出的,从Qt 5.5 开始,您还可以使用QImage::Format_Grayscale8 格式:

// From Qt 5.5
QImage image(inMat.data, inMat.cols, inMat.rows, 
             static_cast<int>(inMat.step),
             QImage::Format_Grayscale8);

通常,您可以将所有MatQImage 的转换包装在一个函数中。下面是cvMatToQImage 最初发现的here 的错误修正更新 版本。

然后,您可以从您的代码中删除所有对QImage 的转换,并改用此函数。

QImage cvMatToQImage(const cv::Mat &inMat)

    switch (inMat.type())
    
        // 8-bit, 4 channel
    case CV_8UC4:
    
        QImage image(inMat.data,
            inMat.cols, inMat.rows,
            static_cast<int>(inMat.step),
            QImage::Format_ARGB32);

        return image;
    

    // 8-bit, 3 channel
    case CV_8UC3:
    
        QImage image(inMat.data,
            inMat.cols, inMat.rows,
            static_cast<int>(inMat.step),
            QImage::Format_RGB888);

        return image.rgbSwapped();
    

    // 8-bit, 1 channel
    case CV_8UC1:
    
#if QT_VERSION >= 0x050500

        // From Qt 5.5
        QImage image(inMat.data, inMat.cols, inMat.rows, 
                     static_cast<int>(inMat.step),
                     QImage::Format_Grayscale8);
#else
        static QVector<QRgb>  sColorTable;

        // only create our color table the first time
        if (sColorTable.isEmpty())
        
            sColorTable.resize(256);
            for (int i = 0; i < 256; ++i)
            
                sColorTable[i] = qRgb(i, i, i);
            
        

        QImage image(inMat.data,
            inMat.cols, inMat.rows,
            static_cast<int>(inMat.step),
            QImage::Format_Indexed8);

        image.setColorTable(sColorTable);
#endif
    

    default:
        qWarning() << "cvMatToQImage() - cv::Mat image type not handled in switch:" << inMat.type();
        break;
    

    return QImage();

【讨论】:

感谢@Miki 提供的这个附加功能 @Optimus1072 很高兴它有帮助。我更新了问题的标题以更好地反映问题并帮助其他人找到此解决方案 对于灰度图像,我们还需要一个 colorMap 对吧? 适用于灰度图像 对于灰度,在 Qt>=5.5 上,您可以使用 QImage::Format_Grayscale8 而不是索引格式。

以上是关于如何将灰度/二进制 Mat 转换为 QImage?的主要内容,如果未能解决你的问题,请参考以下文章

将 cv::mat 转换为 QImage

使用 QImage::loadFromData 将 cv::mat 转换为 QImage

使用原始数据将 QImage 转换为 cv::Mat

cv::Mat转换为QImage错误

Mat与QImage互相转换

Qimage 到 cv::Mat 转换的奇怪行为