QT中對Mat類的一些操作詳解
一、類型轉換
opencv在QT中的應用通常會涉及到這三者的轉換,即Mat、QImage、QPixmap。
下面分別給出瞭
- 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操作內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- OpenCV+Qt實現圖像處理操作
- OpenCV+Qt實現圖像處理操作工具的示例代碼
- Qt實現字符串生成二維碼功能
- VS2017+Qt5+Opencv3.4調用攝像頭拍照並存儲
- 關於PyQt5中QtGui.QImage圖片顯示問題解析