首页 > 编程语言 >基于qml创建最简单的图像处理程序(2)-使用c++&qml进行图像处理

基于qml创建最简单的图像处理程序(2)-使用c++&qml进行图像处理

时间:2022-12-25 10:35:15浏览次数:44  
标签:int image c++ color 图像处理 qml ImageProcessor

 《基于qml创建最简单的图像处理程序》系列课程及配套代码
基于qml创建最简单的图像处理程序(1)-基于qml创建界面

课程1附件

基于qml创建最简单的图像处理程序(2)-使用c++&qml进行图像处理

课程2附件

基于qml创建最简单的图像处理程序(3)-使用opencv&qml进行图像处理

课程3附件


 



            qml实现了不错的界面,但是图像处理这块不是它强项,它能够提供的就是qimage这样的图像,对像素的处理就是对图像的处理,而最简单、最直接的方法就是使用c++的代码进行像素处理。



        这里就涉及到 c++&qml 联合程序设计的具体内容了。这方面涉及到的东西有点多,首先我们需要解决的是qml和c++的联合问题。当然我们在搭建这个框架的时候,可能会有比较多一些的问题—另一个方面,当框架搭建好了以后,使用起来会比较方便。



一、编写按钮触发事件



         现有的例子,已经实现了qml的界面,并且能够打开、显示一个新的图像;我们甚至为后面的图像处理预留了几个按钮的位置,下面我们首先就是要触发这些按钮的事件。



//灰度效果d

Button {

text
:
"灰度";

style
: btnStyle;

onPressedChanged
: {

busy.running
=
true;


// processor.process(fileDialog.fileUrl, ImageProcessor.Gray);

}

}


//浮雕效果

Button {

text
:
"浮雕";

style
: btnStyle;

onPressedChanged
: {

busy.running
=
true;


// processor.process(fileDialog.fileUrl, ImageProcessor.Emboss);

}

}


//黑白效果

Button {

text
:
"黑白";

style
: btnStyle;

onPressedChanged
: {

busy.running
=
true;


// processor.process(fileDialog.fileUrl, ImageProcessor.Binarize);

}

}


   



这个地方有一个小细节,那就是



onPressedChanged()





onClick()



的区别。这两者表示的都是“按下”这个事件,他们的不同在于onClick()要多了一个“按下后抬起”的动作。所以一般来说,对于桌面运用,click用的多一点,而对于Android, onPressedChanged多一点。



 



  二、触发图像处理算法



         我们可以看到上面代码中,每一种处理后面,都有一行注释,调用的是processor对象的一个函数,那么这个函数是从哪里来的了?它的定义来自这里



 



//需要特别注意,这个组件来自imageprocessor这个类,通过这种方法进行集成

ImageProcessor {

id
: processor;

onFinished
: {

imageViewer.source
=
"file:///"
+newFile;

}

}



它的意思是processor对象是ImageProcessor类的一个实例,那么ImageProcessor又是从哪里来的了?



它的定义来自于




基于qml创建最简单的图像处理程序(2)-使用c++&qml进行图像处理_sed


//将c++中定义的类给接过来,在这个定义中,ImageProcessor是c++中函数名,GO.ImageProcessor是你在qml中使用的名称

qmlRegisterType <ImageProcessor
>(
"GO.ImageProcessor",
1,
0,
"ImageProcessor");


 


其中qmlRegisterType是类的引入,类似dllimport之类,而


< ImageProcessor >



"ImageProcessor"


一般来说,都是来自于你的c++的函数名称;而


"GO.ImageProcessor"


是你在qml中的头文件.


 


此外,1,0是你的版本号。


 


所以这个地方有两个头文件,在main.cpp中,引入


 


# include "imageProcessor.h"


而在 qml中,这样引入


 


import GO.ImageProcessor 1. 0


 


那么我们在这里主要讨论的就是如何从c++中将行数引接过来的。


 


 三 、C++中具体算法实现


         那我们就来看c++类中的 ImageProcessor  是如何实现的,它的代码地址来自:



基于qml创建最简单的图像处理程序(2)-使用c++&qml进行图像处理_sed_02


 


 


它的原型为:


void process(QString sourceFile, ImageProcessor      :      :ImageAlgorithm algorithm)

{
QFileInfo fi(sourceFile);
QString destFile = QString( "%1/%2_%3").arg(m_tempPath).arg((
int)algorithm).arg(fi.fileName());
AlgorithmRunnable *r =
new AlgorithmRunnable(sourceFile,destFile,algorithm, this);
m_runnables.append(r);
r - >setAutoDelete(
false);
QThreadPool : :globalInstance()
-
>start(r);

}

 


它设定了一系列参数,主要是创建了AlgorithmRunnalbe,然后通过 QThreadPool 打开新线程运行算法。那么主要的函数体在 AlgorithmRunnable 中的。由于涉及到较多的具体内容,这里不展开说明,大家可以参考全部代码。


我们关注的是图像处理函数,在这个项目中实现了以下内容:

//具体的图像处理算法,注意图片处理的结果直接保存到了destFile中去//

static void _gray(QString sourceFile, QString destFile)
{
QImage image(sourceFile);
if(image.isNull())
{
qDebug() << "load "
<< sourceFile
<<
" failed! ";

return;
}
qDebug() << "depth - "
<< image.depth();


int width = image.width();
int height = image.height();
QRgb color;
int gray;
for( int i = 0; i
< width; i
++)

{
for( int j = 0; j
< height; j
++)

{
color = image.pixel(i, j);
gray = qGray(color);
image.setPixel(i, j, qRgba(gray, gray, gray, qAlpha(color)));
}
}

image.save(destFile);
}

static void _binarize(QString sourceFile, QString destFile)
{
QImage image(sourceFile);
if(image.isNull())
{
qDebug() << "load "
<< sourceFile
<<
" failed! ";

return;
}
int width = image.width();
int height = image.height();
QRgb color;
QRgb avg;
QRgb black = qRgb( 0,
0,
0);

QRgb white = qRgb( 255,
255,
255);

for( int i = 0; i
< width; i
++)

{
for( int j = 0; j
< height; j
++)

{
color = image.pixel(i, j);
avg = (qRed(color) + qGreen(color)
+ qBlue(color))
/
3;

image.setPixel(i, j, avg > =
128
? white
: black);

}
}
image.save(destFile);
}

static void _emboss(QString sourceFile, QString destFile)
{
QImage image(sourceFile);
if(image.isNull())
{
qDebug() << "load "
<< sourceFile
<<
" failed! ";

return;
}
int width = image.width();
int height = image.height();
QRgb color;
QRgb preColor = 0;

QRgb newColor;
int gray, r, g, b, a;
for( int i = 0; i
< width; i
++)

{
for( int j = 0; j
< height; j
++)

{
color = image.pixel(i, j);
r = qRed(color) - qRed(preColor)
+
128;

g = qGreen(color) - qGreen(preColor)
+
128;

b = qBlue(color) - qBlue(preColor)
+
128;

a = qAlpha(color);
gray = qGray(r, g, b);
newColor = qRgba(gray, gray, gray, a);
image.setPixel(i, j, newColor);
preColor = newColor;
}
}
image.save(destFile);
}

//END 具体的图像处理算法,注意图片处理的结果直接保存到了destFile中去//


 


 


甚至不用去看算法的实现(因为最后我们肯定是要使用OpenCV来做图像处理的),这3个图像处理的函数都是非常统一的


 


static      void _binarize(QString sourceFile, QString destFile)

比如返回类型为void,第一个参数为输入图像,第二个参数为输出图片。从这里也可以看出,在我们这个项目中,是通过地址(而不是内存)来传递数据的。如果我们想要添加新的算法,修改现有算法,直接修改这几个函数、添加相关界面就可以。


 



基于qml创建最简单的图像处理程序(2)-使用c++&qml进行图像处理_c++_03


而下一步,最关键的一步,也是原书例子没有实现的,就是添加OpenCV的代码,使用它来替代进行图像处理。


 

标签:int,image,c++,color,图像处理,qml,ImageProcessor
From: https://blog.51cto.com/jsxyhelu2017/5967923

相关文章

  • 基于qml创建最简单的图像处理程序(3)-使用opencv&qml进行图像处理
    《基于qml创建最简单的图像处理程序》系列课程及配套代码基于qml创建最简单的图像处理程序(1)-基于qml创建界面(0)​​课程1附件基于qml创建最简单的图像处理程序(2)-使用c++&qm......
  • C++ empty函数
    https://blog.csdn.net/qq_41598072/article/details/99973908empty是用来测试变量是否已经配置。若变量已存在、非空字符串或者非零,则返回false值;反之返回true值。所......
  • C++ sort函数中利用lambda进行自定义排序规则
    在c++中,由于sort()函数默认提供的是由小到大的排序方式,因此有时候我们需要自定义排序规则来实现由大到小的排序。一维vector<>排序#include<bits/stdc++.h>usingnam......
  • C++:重载运算符
    基本概念通常我们自定义的类类型,不具有内置类型的一些操作,比如int类型的算术运算,指针类型的解引用、取地址操作,容器类型的下标操作等。因此,如果希望我们自定义的类类型......
  • C++科研人员信息管理系统
    C++科研人员信息管理系统某科研团队主要有四类人员:科研主管、研究员、研究助理和实习研究员。现在,需要存储这些人员的姓名、编号、级别、当月薪水,计算月薪总额并显示全部......
  • C/C++在线餐馆预订管理系统
    C/C++在线餐馆预订管理系统软件学院实训任务书一、实训名称实践环节数据结构与算法实训项目名称在线餐馆预订管理系统二、学生信息班级......
  • c++ 模板参数有默认值时模板特例化匹配问题
    如下的源码:template<typenameT,typenameU=int>classS{//#1public:voidf1(){};};template<>classS<void>{ //#2public:voidf2(){};};......
  • 蓝桥-13届-C++-B组-省赛-F题-统计子矩阵
    直达链接主要解题思路分为两个部分,1是构造二维前缀和计算矩阵和,降低每次求和的时间复杂度;2是对所有子矩阵的遍历求和过程,因为需要两个坐标,遍历4个行/列值,4层for循环时间复......
  • C++基础3
    C++基础3typedef为现有类型创建一个新名字主要有以下几种形式:为基本数据类型定义别名为指针定义别名为自定义数据类型定义别名为数组定义别名声明函数定义新......
  • C/C++编译器配置——MinGW下载安装
    C/C++编译器配置——MinGW下载安装前言本文主要讲述如何安装C语言编译器——MinGW,特点是文章附有完整详细的实际安装过程截图,文字反而起说明提示作用。编写本文的原因......