配置环境:qt5.2,opencv 2.4.8,vs2010,win8.1 32位


注:读取图片的路径一定不要包含中文



Qt是跨平台C++图形用户界面应用程序开发框架。主要包括以下几个部分:(1)跨平台的IDE,被称为Qt Creater;(2)一系列Qt类库和开发工具。使用Qt SDK 开发C++应用有很多好处:



  1. Qt团队开发的开源代码,让你接触不同Qt组件的源代码、。
  2. 跨平台,意味着开一在不同的操作系统上开发,比如Windows,Linux,Mac OS X等。
  3. 包括了完整的跨平台GUI库,提供有效的面向对象和事件驱动的模型。
  4. Qt包括不同的跨平台库来发展多媒体、图形、数据库,多线程,web应用以及其对先进设计有用的模块。


上面这段话是从英文上翻译过来的,主要就是说可以跨多个平台。



尽管之前看了浅墨的文章,对于opencv的内容不再陌生,但是qt依然是自己的弱点。这篇文章就归类在qt里面了。因为我觉得,Qt的问题比较多,而涉及opencv的内容比较少。



这次要做的就是实现从本地读取图像,在Qt设计师中显示。然后反转再现实。打开图像显示如下:

QT与opencv编程_win8



点击处理图像之后,显示如下:

QT与opencv编程_win8_02


首先建立一个qt解决方案,命名为 imagprocess

步骤如下:

1、在Qt设计师中选择push_botton。显示的名字分别显示为打开图像、处理图像,objective_name分别为 open,process.

QT与opencv编程_win8_03



2、梳理下本文的思路:点击打开图像,要在本地选择一张图像并输出;点击处理图像,图像被旋转,然后输出。主要就是为了实现这两个内容。

声明:由于在自己学习的时候,遇到了两个版本,一个简单,一个复杂。所以说,这里将会分成两块来说。为了不显得太复杂,就不用加标注的方法来说,就单独来说。

简单的显示:

3.1、imagprocess.h内容填写,加粗的部分是需要我们填写的。在类的声明中,需要将用到的声明都写清楚,便于后续的使用。

#ifndefIMAGPROCESS_H
#defineIMAGPROCESS_H
 #include<QtWidguets/QMainWindow>
#include"ui_imagprocess.h"
#include <QFileDialog.h>//getOpenFileName的类声明
#include <opencv2/core/core.hpp>
#include<opencv2/highgui/highgui.hpp>//常规的opencv声明
classimagprocess : public QMainWindow
{
      Q_OBJECT
public:
     imagprocess(QWidget *parent = 0);
      ~imagprocess();
private slots:
      voidon_open_clicked();
     voidon_process_clicked();//声明两个槽函数,分别对应open、process
 private:
     Ui::imagprocessClass ui;
    cv::Mat image; // the image variable
};
#endif// IMAGPROCESS_H

总的来说,没有需要特别说明的。这些都是比较基本的,当然了在不知情的情况下。总是会出现很多的问题,这些都是在不断地阅读中明白的。没有什么文献可以阅读,书籍去约定俗称。遇到了,记下来,然后就可以了。

4.1、imagprocess.cpp的填写。这个是关键,所以出现了很多问题。也算是写了很长时间。所以说,这里将会把自己遇到的众多问题跟大家分享一下。

#include"imagprocess.h"
using namespace cv;
imagprocess::imagprocess(QWidget*parent)
      : QMainWindow(parent)
{
     ui.setupUi(this);
      //connect(ui.open,SIGNAL(clicked()),this,SLOT(on_open_clicked()));
//connect(ui.process,SIGNAL(clicked()),this,SLOT(on_process_clicked()));
}
imagprocess::~imagprocess()
{
}

void imagprocess::on_open_clicked()
{
    QString fileName =QFileDialog::getOpenFileName(this, tr("Open Image"),  
    ".", tr("Image Files (*.png *.jpg *.jpeg*.bmp)")); 
  std::string str = fileName.toStdString();
   image=imread(str);
//image = cv::imread(fileName.toLatin1().data());
  namedWindow("Original Image");
  imshow("Original Image",image); 
}
 void imagprocess::on_process_clicked()
{
     flip(image, image, 1);    
     namedWindow("Output Image");   
    imshow("Output Image", image);
}

这是函数最核心的部分。将会详细去说:

(1)//connect(ui.open,SIGNAL(clicked()),this,SLOT(on_open_clicked()));
/connect(ui.process,SIGNAL(clicked()),this,SLOT(on_process_clicked()));

这个是槽函数的连接函数,如果在Qt设计师里没有添加槽函数的话,是需要用语句来写出来的。当然,程序运行了之后,我发现打开图像要进行两次,所以就想着把它注释掉,结果不影响运行。所以,这里我就比较疑惑。

当我还没写完的时候,问题就解决了:由于我在声明函数的过程中,使用了on_ 这样就有内置的函数,来完成链接。所以问题就变得更加简单了!

(2)

void imagprocess::on_open_clicked()
{
  QString fileName =QFileDialog::getOpenFileName(this, tr("Open Image"),  
   ".",tr("Image Files (*.png*.jpg *.jpeg *.bmp)")); 
 std::string str = fileName.toStdString();
  image=imread(str);
  //image =cv::imread(fileName.toLatin1().data());
 namedWindow("Original Image");
 imshow("Original Image",image); 
}

函数on_open_clicked()的主体,当点击打开图像之后,就会执行这里的内容。当然,为了简单,并没有对异常情况进行判断。这个之后会慢慢来弥补。

QStringfileName = QFileDialog::getOpenFileName(this, tr("Open Image"),  
 ".", tr("Image Files (*.png*.jpg *.jpeg *.bmp)"));

获取图像的路径,比如E:\cat.jpg等。这个没什么好说的,很明了,就是自己去写的时候,需要多加注意。每个参数的意思也很明确。

std::stringstr = fileName.toStdString();
image=imread(str);
//image= cv::imread(fileName.toLatin1().data());

这三行内容放在一起说。前两行所要表达的内容和第三行是一样的。上一行中,fileName是Qstring类型的,也就说想要利用cv::imread来显示不可能的。为了用imread来显示,先来看看imread的结构:

Matimaread(const string &filename,intflags=1)这里的filename要求是string来显示。所以就要转换了,第一行就是将Qstring转换成string.

第三行就直接可以了。

namedWindow("OriginalImage");
imshow("OriginalImage",image);

参考opencv的内容,本人已经烂熟于心,所以不多说。

(3)

void imagprocess::on_process_clicked()
{
   flip(image, image, 1);   
  namedWindow("Output Image");   
   imshow("Output Image", image);
}

点击process,将会执行的函数内容。flip就是一个图像的翻转,没什么难度

void flip(InputArray src, OutputArray dst, int flipCode);



参数fipCode:整数,水平发转;0垂直反转;负数,水平垂直均反转。就是这样了。

Main函数不做处理就是了。

5、实验结果

void flip(InputArray src, OutputArray dst, int flip

QT与opencv编程_win8_04


没什么问题哦!当然了,没有充分利用到qt,只是用了两个按钮。不过,这也让我们体验一下图像显示的小成就。

 

稍微复杂点的显示


 3.2 imagprocess.h


#ifndef IMAGPROCESS_H
#define IMAGPROCESS_H 
#include <QtWidgets/QMainWindow>
#include "ui_imagprocess.h"
#include<QFileDialog.h>
#include<Qlabel.h>//label的声明
#include<opencv/cv.h>//后面用的cvtColor的声明
#include<opencv2/core/core.hpp>
#include<opencv2/highgui/highgui.hpp>
class imagprocess : public QMainWindow
{
      Q_OBJECT
public:
     imagprocess(QWidget*parent = 0);
      ~imagprocess();
 private slots:
      void on_open_clicked();
    void on_process_clicked();
private:
     Ui::imagprocessClassui;
     cv::Mat image; // the image variable
};

 #endif // IMAGPROCESS_H

其他的内容跟简单的版本没有差别。

4.2 imagprocess.cpp

#include "imagprocess.h"
using namespace cv;
imagprocess::imagprocess(QWidget *parent)
      :QMainWindow(parent)
{
      ui.setupUi(this);
      //connect(ui.open,SIGNAL(clicked()),this,SLOT(on_open_clicked()));
   //connect(ui.process,SIGNAL(clicked()),this,SLOT(on_process_clicked()));
}
imagprocess::~imagprocess()
{
}
voidimagprocess::on_open_clicked()
{
QString fileName =QFileDialog::getOpenFileName(this, tr("Open Image"),  
 ".", tr("Image Files (*.png*.jpg *.jpeg *.bmp)")); 
 //std::string str = fileName.toStdString();
 //image=imread(str);
 image =cv::imread(fileName.toLatin1().data());
 cvtColor(image,image,CV_RGB2RGBA); 
  QImage img = QImage((const unsignedchar*)(image.data), image.cols, image.rows, QImage::Format_RGB32); 
  QLabel *label = new QLabel(this); 
  label->move(100, 20);//图像在窗口中所处的位置; 
   label->setPixmap(QPixmap::fromImage(img)); 
  label->resize(label->pixmap()->size());     
   label->show();
}
voidimagprocess::on_process_clicked()
{
    cv::flip(image, image, 1);   
    cv::cvtColor(image, image,CV_RGB2RGBA); 
       QImage img = QImage((const unsignedchar*)(image.data), image.cols, image.rows, QImage::Format_RGB32); 
       QLabel *label = new QLabel(this); 
       label->move(100, 20); 
       label->setPixmap(QPixmap::fromImage(img)); 
       label->resize(label->pixmap()->size());     
       label->show();
}


程序解释:

voidimagprocess::on_open_clicked()里面的内容:

(1)cvtColor(image,image,CV_RGB2RGBA); 

cvtColor是Opencv里的颜色空间转换函数,可以实现RGB颜色向HSV,HSI等颜色空间的转换,也可以转换成灰度图像。

void cvCvtColor( const CvArr*src, CvArr* dst, int code );

第一个参数是输入图像

第二个参数是输出图像

第三个参数是彩色空间转换的模式,能实现不同类型的颜色空间转换。比如CV_BGR2GRAY表示转换为灰度图,CV_BGR2HSV将图片从RGB空间转换为HSV空间。当code选用CV_BGR2HSV时,对于8位图,需要将RGB值归一化到0-1之间。这样得到HSV图中的H范围才是0-360,S和V的范围是0-1。图像在QT显示前,必须转化成QImage格式,图像在QT显示前,必须转化成QImage格式,将RGBA格式转化成RGB。到底是什么格式转换成什么呢?这个有点疑惑,为什么成了RGBA转换成RGB了呢。这个也算是个遗留的问题吧。

(2)QImage img = QImage((const unsigned char*)(image.data),image.cols, image.rows, QImage::Format_RGB32); 

这句是将cv::image转换成QImage。

(3)QLabel *label = new QLabel(this); 

QT平台下创建一个标签的语句。

(4) label->move(100, 20); /

点的位置是(100,20),图像的左上角坐标

(5)label->setPixmap(QPixmap::fromImage(img)); 

就是将Qimage显示,QPixmap的具体信息,还没有搞明白。

(6)label->resize(label->pixmap()->size());   

更改尺寸  

(7)label->show();

最后显示。从(3)-(7)是QT图象显示的部分,还不算很明白。不过先记住!

on_process_clicked()函数就不多介绍了。

显示一个效果吧。分别是原始图像和反转图像。很显然,颜色有变化。那就是cvtColor的效果吧。不过,还是希望显示一个正常的颜色了。具体要怎么弄呢?以后再做研究。

QT与opencv编程_win8

QT与opencv编程_win8_02




注:写程序的过程中遇到的问题。

1、class “QString”没有成员“toAscii”

QT与opencv编程_QT与opencv编程_07

这是在寻找别人程序的过程中看到的。然后就出现了这样的问题。然后将toAscii换成toLatin1就OK了。这是由于版本升级导致的。2.4版本以后就不用toAscii了。

2、Debud Error!R6010-abort() has been called

QT与opencv编程_qt_08


这个问题太常见了。主要问题是:下标等越界,指针位置不恰当。而这里呢,是由于我犯了一个错误。就是在读取图片的过程中,路径是E:\C++程序\cat.jpg里面包含了汉字。然后就出现了这个问题。个人认为,这样导致了fileName无法正确被使用,所以读取图片自己让对不对了。

因此,不管是什么语言编程,除了注释,方便自己去理解,一定要用英文的。这样才不会因为这些细小的问题,浪费了自己的时间。