基于QT實現(xiàn)顯示OpenCV讀取的圖片
1. 概述
OpenCV自帶了一部分常用的GUI功能,但是更多的圖像處理功能需要其他GUI框架來輔助實現(xiàn),這里通過QT來顯示OpenCV讀取的圖片。
2. 實現(xiàn)
在QtCreator中新建一個基于QMainWindow的應(yīng)用:
其中QImageShowWidget就是用于顯示圖像的控件,它是繼承于QWidget實現(xiàn)的,可以將其嵌入QMainWindow的centralwidget中:
QImageShowWidget是自定義的顯示組件,可以首先在QtCreator的設(shè)計師界面拖入一個QWidget,再通過“窗口部件提升”功能提升為QImageShowWidget。
2.1 代碼
qimageshowwidget.h代碼如下:
#ifndef QIMAGESHOWWIDGET_H #define QIMAGESHOWWIDGET_H #include <QWidget> class QImageShowWidget : public QWidget { Q_OBJECT public: explicit QImageShowWidget(QWidget *parent = nullptr); ~QImageShowWidget(); bool LoadImage(const char* imagePath); signals: public slots: protected: void paintEvent(QPaintEvent *); //繪制 void Release(); private: uchar* winBuf; //窗口填充buf int winWidth; //窗口像素寬 int winHeight; //窗口像素高 int winBandNum; //波段數(shù) }; #endif // QIMAGESHOWWIDGET_H
qimageshowwidget.cpp代碼如下:
#include "qimageshowwidget.h" #include <opencv2\opencv.hpp> #include <QPainter> #include <QDebug> #include <iostream> using namespace cv; using namespace std; QImageShowWidget::QImageShowWidget(QWidget *parent) : QWidget(parent) { //填充背景色 setAutoFillBackground(true); setBackgroundRole(QPalette::Base); winBuf = nullptr; winWidth = rect().width(); winHeight = rect().height(); winBandNum = 3; } QImageShowWidget::~QImageShowWidget() { if(winBuf) { delete[] winBuf; winBuf = nullptr; } } bool QImageShowWidget::LoadImage(const char* imagePath) { //從文件中讀取成灰度圖像 Mat img = imread(imagePath); if (img.empty()) { fprintf(stderr, "Can not load image %s\n", imagePath); return false; } Release(); winWidth = rect().width(); winHeight = rect().height(); size_t winBufNum = (size_t) winWidth * winHeight * winBandNum; winBuf = new uchar[winBufNum]; memset(winBuf, 255, winBufNum*sizeof(uchar)); for (int ri = 0; ri < img.rows; ++ri) { for (int ci = 0; ci < img.cols; ++ci) { for(int bi = 0; bi < winBandNum; bi++) { size_t m = (size_t) winWidth * winBandNum * ri + winBandNum * ci + bi; size_t n = (size_t) img.cols * winBandNum * ri + winBandNum * ci + bi; winBuf[m] = img.data[n]; } } } update(); return true; } //重新實現(xiàn)paintEvent void QImageShowWidget::paintEvent(QPaintEvent *) { if(!winBuf) { return; } QImage::Format imgFomat = QImage::Format_RGB888; QPainter painter(this); QImage qImg(winBuf, winWidth, winHeight, winWidth*winBandNum, imgFomat); painter.drawPixmap(0, 0, QPixmap::fromImage(qImg)); } void QImageShowWidget::Release() { if(winBuf) { delete[] winBuf; winBuf = nullptr; } }
2.2 解析
所有基于QWidget的類都可以重新實現(xiàn)界面重繪事件paintEvent(),它會在界面需要的時候(例如調(diào)用update())自動重繪。在這個事件函數(shù)中可以通過圖形繪制接口QPainter繪制:
QImage::Format imgFomat = QImage::Format_RGB888; QPainter painter(this); QImage qImg(winBuf, winWidth, winHeight, winWidth*winBandNum, imgFomat); painter.drawPixmap(0, 0, QPixmap::fromImage(qImg));
可以看到QPainter繪制的其實是QImage對象,也就是重點是構(gòu)造QImage這個對象。這個對象是由申請的內(nèi)存winBuf來構(gòu)建的。顯示的圖像是由寬、高以及波段組成的,需要將三維空間壓縮為一維空間——簡單來講,內(nèi)存的組成為RGBRGBRGB...,并且起點位置為左上角,由左至右,由上至下。
OpenCV讀取的圖像為Mat對象:
//從文件中讀取成灰度圖像 Mat img = imread(imagePath); if (img.empty()) { fprintf(stderr, "Can not load image %s\n", imagePath); return false; }
Mat對象可以通過data()方法直接訪問讀取的圖像內(nèi)存。而這塊內(nèi)存也是RGBRGBRGB...的結(jié)構(gòu)組成,并且起點位置也是左上角,由左至右,由上至下。將其逐像素傳入到申請的內(nèi)存winBuf:
winWidth = rect().width(); winHeight = rect().height(); size_t winBufNum = (size_t) winWidth * winHeight * winBandNum; winBuf = new uchar[winBufNum]; memset(winBuf, 255, winBufNum*sizeof(uchar)); for (int ri = 0; ri < img.rows; ++ri) { for (int ci = 0; ci < img.cols; ++ci) { for(int bi = 0; bi < winBandNum; bi++) { size_t m = (size_t) winWidth * winBandNum * ri + winBandNum * ci + bi; size_t n = (size_t) img.cols * winBandNum * ri + winBandNum * ci + bi; winBuf[m] = img.data[n]; } } }
3. 結(jié)果
通過界面加載一張圖像,顯示結(jié)果如下:
到此這篇關(guān)于基于QT實現(xiàn)顯示OpenCV讀取的圖片的文章就介紹到這了,更多相關(guān)QT顯示OpenCV讀取圖片內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++ map 根據(jù)value找key的實現(xiàn)
今天小編就為大家分享一篇C++ map 根據(jù)value找key的實現(xiàn),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-12-12C++常用函數(shù)總結(jié)(algorithm 頭文件)
本文給大家詳細(xì)介紹了algorithm 頭文件中最常用的函數(shù)及其使用方法,當(dāng)然這只是其中的一部分,algorithm 頭文件中還有很多其他的函數(shù),感興趣的朋友一起看看吧2023-12-12