首頁 > 軟體

QT中對Mat類的一些操作詳解

2022-06-08 22:02:38

一、型別轉換

opencv在QT中的應用通常會涉及到這三者的轉換,即MatQImageQPixmap
下面分別給出了

  • Mat轉QImage
  • QImage轉Mat
  • Mat轉QPixmap

1️⃣:Mat轉QImage

QImage MainWindow::MatToImage(const Mat &m)  //Mat轉Image
{
    switch(m.type())
    {
        case CV_8UC1:
        {
            QImage img((uchar *)m.data,m.cols,m.rows,m.cols * 1, QImage::Format_Grayscale8);
            return img;
        }
            break;
        case CV_8UC3:
        {
            QImage img((uchar *)m.data,m.cols,m.rows,m.cols * 3, QImage::Format_RGB888);
            return img.rgbSwapped();  //因為在QT中彩色圖象是RGB的順序,但是在OPENCV中是BGR的順序,所以要轉一下
        }
            break;
        case CV_8UC4:
        {
            QImage img((uchar *)m.data,m.cols,m.rows,m.cols * 4, QImage::Format_ARGB32);
            return img;
        }
            break;
        default:     //如果是預設的,那麼將其返回為一個空物件
        {
            QImage img;
            return img;
        }
    }
}

2️⃣:QImage轉Mat

Mat MainWindow::ImageToMat(const QImage &img,bool inCloneImageData)  //Image轉Mat
{
    switch(img.format())
    {
        case QImage::Format_Indexed8:   //單通道
        {
            Mat  mat( img.height(), img.width(), CV_8UC1,
                          const_cast<uchar*>(img.bits()), static_cast<size_t>(img.bytesPerLine()) );

            return (inCloneImageData ? mat.clone() : mat);
        }
        // 8-bit, 3 通道
        case QImage::Format_RGB32:   //這種寫法表示並列關係
        case QImage::Format_RGB888:
        {
            if ( !inCloneImageData )
            {
                qWarning() << "CVS::QImageToCvMat() - Conversion requires cloning because we use a temporary QImage";
            }

            QImage  swapped = img;

            if ( img.format() == QImage::Format_RGB32 )
            {
                swapped = swapped.convertToFormat( QImage::Format_RGB888 );
            }

            swapped = swapped.rgbSwapped();  //因為在QT中彩色圖象是RGB的順序,但是在OPENCV中是BGR的順序,所以要轉一下

            return Mat( swapped.height(), swapped.width(), CV_8UC3,
                        const_cast<uchar*>(swapped.bits()), static_cast<size_t>(swapped.bytesPerLine()) ).clone();
        }
        // 8-bit, 4 channel
        case QImage::Format_ARGB32:
        case QImage::Format_ARGB32_Premultiplied:
        {
            Mat  mat( img.height(), img.width(), CV_8UC4,
                          const_cast<uchar*>(img.bits()), static_cast<size_t>(img.bytesPerLine()) );

            return (inCloneImageData ? mat.clone() : mat);
        }

        // 8-bit, 1 channel
        default:
            qWarning() << "CVS::QImageToCvMat() - QImage format not handled in switch:" << img.format();
            break;
        }
    return Mat();
}

3️⃣:Mat轉QPixmap

QPixmap MainWindow::MatToPixmap(const Mat &m)
{
    return QPixmap::fromImage(MatToImage(m));   //相當於先將Mat轉成Image,再轉成Pixmap
}

二、儲存至資料庫

我這裡是直接將Mat型別的資料以二進位制資料流的方式儲存到資料庫中,有些文章是將檔名及其所在的路徑儲存到資料庫中,這個還是要好看專案需求,個人而言,我這個要更復雜一點。
演示效果如下:

1️⃣:基礎介面

2️⃣:磨皮處理

3️⃣:資料庫中的資料

4️⃣:核心程式碼

注意: 建立資料表的時候,欄位的型別,一定要滿足資料的大小,比方說儲存圖片一般使用blob相關的型別,其中blob最大為64k,mediumblob最大為16M。

思路,先捕獲ui控制元件(我用於顯示圖片的控制元件是QLabel)中的資料,然後進行資料庫操作。具體步驟如下:

資料型別:Image —>Mat。

然後再將Mat型別儲存到byte陣列中,再上傳到資料庫。(此時有小夥伴可能就要問了,為什麼不直接從Image型別轉byte陣列呢?當然可以,不過我們饒了個彎子也是想讓大家學會如何將Mat型別轉成byte陣列)

上傳資料庫,使用者名稱+圖片資料(這個看自身的需求)

準備資料庫查詢語句query,用法如下圖所示

驗證階段,取回圖片,然後顯示在右側的QLabel中,因為我程式碼中取回的是使用者id為33的圖片資料,所以顯示的是一個水瓶。

void MainWindow::Upmysql()   //將處理的圖片上傳至資料庫
{
    QImage pix=ui->Process_image->pixmap()->toImage();
    Mat m=ImageToMat(pix);  //QImage--->Mat格式轉換
    int height = pix.height();   //定義這兩者的目的是為了傳給Mat的建構函式
    int width = pix.width();
    int iSize = m.total() * m.elemSize();   //記錄Mat影象的大小,以便於建立同等大小的位元組陣列
    unsigned char* bytes = new unsigned char[iSize];   //建立一個位元組陣列,用於儲存二進位制資料
    memcpy(bytes, m.data, iSize * sizeof(unsigned char));    //將Mat型別的資料賦給byte陣列
    //qDebug()<<QByteArray((char*)bytes, 100);    //檢視前100個字元
    QByteArray sbuf = QByteArray::fromRawData((char *)bytes, iSize * sizeof(unsigned char));   //將unsigned char轉為QByteArray型別

    QVariant var(sbuf);   //將QByteArray型別轉成QVariant以便於插入到MYSql
    QSqlQuery query;   //下面為資料庫查詢的一種方式,要特別注意格式的要求
    //建立資料表的時候,一定要注意資料的大小,比方說圖片blob為64k,mediumblob為16m
    query.prepare("INSERT INTO Image_All (id, img_data) "
                        "VALUES (:id, :img_data)");
    query.bindValue(":id", 77877);    //我這裡隨便設的
    query.bindValue(":img_data",var);
    if(query.exec())
    {
        qDebug()<<"圖片成功上傳至資料庫";
    }
    else{
        qDebug()<<"圖片上傳資料庫失敗";
    }

    QString sql1=QString("select img_data from Image_All where id='33'");   //獲取資料庫中圖片資料
    if(query.exec(sql1)) //執行sql語句是否成功
    {
        while(query.next())//指向下一條
        {
            //根據下標將返回結果進行分割
            QByteArray TEXT1=query.value(0).toByteArray();  //將查詢結果以QByteArray形式返回
            unsigned char *data2;
             data2 = reinterpret_cast<unsigned char*>(TEXT1.data());   //將QByteArray型別轉為unsigned char
            Mat image=Mat(height,width,CV_8UC3,data2);      //將unsigned char轉成Mat型別的資料
            QPixmap temp = MatToPixmap(image);
            ui->Process_image->setPixmap(temp);
        }

    }else
    {
        qDebug()<<"從資料庫獲取圖片失敗";
    }

}

到此這篇關於QT中對Mat類的一些操作詳解的文章就介紹到這了,更多相關QT Mat操作內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com! 


IT145.com E-mail:sddin#qq.com