Qt下使用QImage和OpenCV实现图像的拼接与融合

文章目录

  • 前言
  • 一、使用QImage进行水平拼接
  • 二、使用OpenCV进行水平拼接
  • 三、使用OpenCV进行图像融合
  • 四、示例完整代码
  • 总结


前言

本文主要讲述了在Qt下使用QImage和OpenCV实现图像的拼接与融合,并结合相应的示例进行讲解,以便大家学习,如有错误之处,欢迎大家批评指正。

项目效果
请添加图片描述


提示:以下是本篇文章正文内容,下面案例可供参考

一、使用QImage进行水平拼接

这里新建了一个QImage对象,然后通过QPainter来将两张图像绘制在一起,并且可以通过QImage的save函数来保存拼接后的图像:
请添加图片描述

//使用QImage进行水平拼接
QPixmap Widget::imageMosaic(const QImage &image1,const QImage &image2)
{
    //检查图像是否有效
    if(image1.isNull() || image2.isNull())
    {
        return QPixmap();
    }

    //计算拼接后的图像尺寸
    int newWidth = image1.width() + image2.width();
    int newHeight = std::max(image1.height(),image2.height());

    //创建一个新的QImage对象
    QImage newImage(newWidth,newHeight,QImage::Format_RGB32);
    newImage.fill(Qt::transparent);   //填充为透明,如果背景不是透明的则可以选择其他颜色

    //使用QPainter来绘制图像
    QPainter painter(&newImage);
    painter.drawImage(0,0,image1);   //在新图像的(0,0)位置绘制第一张图像
    painter.drawImage(image1.width(),0,image2);   //在新图像的(image1.width(),0)位置绘制第二张图像

    //结束绘制
    painter.end();

    //保存拼接图像
    newImage.save("E:/myPhoto/imageMosaic.jpg");

    //返回拼接图像
    return QPixmap::fromImage(newImage);
}

二、使用OpenCV进行水平拼接

在Qt中配置好OpenCV环境后,就可以使用OpenCV中的hconcat函数来进行图像的拼接了,同时可以使用imwrite来保存拼接图像:
请添加图片描述

//使用OpenCV进行水平拼接
QPixmap Widget::opencvMosaic(const cv::Mat &mat1,const cv::Mat &mat2)
{
    //检查图像是否有效
    if(mat1.empty() || mat2.empty())
    {
        return QPixmap();
    }

    //判断两张图像的高度
    if(mat1.rows != mat2.rows)
    {
        return QPixmap();
    }

    //水平拼接图像
    cv::Mat resultMat;
    cv::hconcat(mat1,mat2,resultMat);

    //将OpenCV的Mat转换为QImage
    //cv::cvtColor(resultMat,resultMat,cv::COLOR_BGR2RGB);
    QImage newImage((const unsigned char*)(resultMat.data),resultMat.cols,resultMat.rows,resultMat.step,QImage::Format_RGB888);

    //QImage的save保存拼接图像
    //newImage.save("E:/myPhoto/opencvMosaic.jpg");

    //OpenCV的imwrite保存拼接图像
    cv::imwrite("E:/myPhoto/opencvMosaic.jpg",resultMat);

    //返回拼接图像
    return QPixmap::fromImage(newImage.rgbSwapped());
}

三、使用OpenCV进行图像融合

这个示例的图像融合比较简单,使用了OpenCV中的addWeighted函数来进行融合:
请添加图片描述

//使用OpenCV进行图像融合
QPixmap Widget::opencvMerge(const cv::Mat &mat1,const cv::Mat &mat2)
{
    //检查图像是否有效
    if(mat1.empty() || mat2.empty())
    {
        return QPixmap();
    }

    //图像融合(这里只是一个简单的示例,使用加权平均)
    cv::Mat resultMat;
    double alpha = 0.5;   //融合系数,可以根据需要调整
    cv::addWeighted(mat1,alpha,mat2,1.0 - alpha,0.0,resultMat);

    //将OpenCV的Mat转换为QImage
    //cv::cvtColor(resultMat,resultMat,cv::COLOR_BGR2RGB);
    QImage newImage((const unsigned char*)(resultMat.data),resultMat.cols,resultMat.rows,resultMat.step,QImage::Format_RGB888);

    //QImage的save保存拼接图像
    //newImage.save("E:/myPhoto/opencvMerge.jpg");

    //OpenCV的imwrite保存拼接图像
    cv::imwrite("E:/myPhoto/opencvMerge.jpg",resultMat);

    //返回拼接图像
    return QPixmap::fromImage(newImage.rgbSwapped());

}

四、示例完整代码

1.widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QImage>
#include <QPixmap>
#include <QPainter>
#include <QMessageBox>
#include <QFileDialog>
#include <QDebug>
#include "opencv2/opencv.hpp"

using namespace cv;
using namespace std;

QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();

    QPixmap imageMosaic(const QImage &image1,const QImage &image2);
    QPixmap opencvMosaic(const cv::Mat &mat1,const cv::Mat &mat2);
    QPixmap opencvMerge(const cv::Mat &mat1,const cv::Mat &mat2);

    void setShowImage(int index);

private slots:
    void on_pb_image_1_clicked();
    void on_pb_image_2_clicked();
    void on_pb_imageMosaic_clicked();
    void on_pb_opencvMosaic_clicked();
    void on_pb_opencvMerge_clicked();
    void on_pb_keep_clicked();
    void on_pb_fill_clicked();

private:
    Ui::Widget *ui;

    //QImage对象
    QImage m_image_1;
    QImage m_image_2;
    QPixmap m_showPixmap;

    //CV::Mat对象
    cv::Mat m_mat_1;
    cv::Mat m_mat_2;

};
#endif // WIDGET_H

2.widget.cpp

#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
}

Widget::~Widget()
{
    delete ui;
}

//使用QImage进行水平拼接
QPixmap Widget::imageMosaic(const QImage &image1,const QImage &image2)
{
    //检查图像是否有效
    if(image1.isNull() || image2.isNull())
    {
        return QPixmap();
    }

    //计算拼接后的图像尺寸
    int newWidth = image1.width() + image2.width();
    int newHeight = std::max(image1.height(),image2.height());

    //创建一个新的QImage对象
    QImage newImage(newWidth,newHeight,QImage::Format_RGB32);
    newImage.fill(Qt::transparent);   //填充为透明,如果背景不是透明的则可以选择其他颜色

    //使用QPainter来绘制图像
    QPainter painter(&newImage);
    painter.drawImage(0,0,image1);   //在新图像的(0,0)位置绘制第一张图像
    painter.drawImage(image1.width(),0,image2);   //在新图像的(image1.width(),0)位置绘制第二张图像

    //结束绘制
    painter.end();

    //保存拼接图像
    newImage.save("E:/myPhoto/imageMosaic.jpg");

    //返回拼接图像
    return QPixmap::fromImage(newImage);
}

//使用OpenCV进行水平拼接
QPixmap Widget::opencvMosaic(const cv::Mat &mat1,const cv::Mat &mat2)
{
    //检查图像是否有效
    if(mat1.empty() || mat2.empty())
    {
        return QPixmap();
    }

    //判断两张图像的高度
    if(mat1.rows != mat2.rows)
    {
        return QPixmap();
    }

    //水平拼接图像
    cv::Mat resultMat;
    cv::hconcat(mat1,mat2,resultMat);

    //将OpenCV的Mat转换为QImage
    //cv::cvtColor(resultMat,resultMat,cv::COLOR_BGR2RGB);
    QImage newImage((const unsigned char*)(resultMat.data),resultMat.cols,resultMat.rows,resultMat.step,QImage::Format_RGB888);

    //QImage的save保存拼接图像
    //newImage.save("E:/myPhoto/opencvMosaic.jpg");

    //OpenCV的imwrite保存拼接图像
    cv::imwrite("E:/myPhoto/opencvMosaic.jpg",resultMat);

    //返回拼接图像
    return QPixmap::fromImage(newImage.rgbSwapped());
}

//使用OpenCV进行图像融合
QPixmap Widget::opencvMerge(const cv::Mat &mat1,const cv::Mat &mat2)
{
    //检查图像是否有效
    if(mat1.empty() || mat2.empty())
    {
        return QPixmap();
    }

    //图像融合(这里只是一个简单的示例,使用加权平均)
    cv::Mat resultMat;
    double alpha = 0.5;   //融合系数,可以根据需要调整
    cv::addWeighted(mat1,alpha,mat2,1.0 - alpha,0.0,resultMat);

    //将OpenCV的Mat转换为QImage
    //cv::cvtColor(resultMat,resultMat,cv::COLOR_BGR2RGB);
    QImage newImage((const unsigned char*)(resultMat.data),resultMat.cols,resultMat.rows,resultMat.step,QImage::Format_RGB888);

    //QImage的save保存拼接图像
    //newImage.save("E:/myPhoto/opencvMerge.jpg");

    //OpenCV的imwrite保存拼接图像
    cv::imwrite("E:/myPhoto/opencvMerge.jpg",resultMat);

    //返回拼接图像
    return QPixmap::fromImage(newImage.rgbSwapped());

}

//设置显示图像
void Widget::setShowImage(int index)
{
    QPixmap showPixmap;
    if(index == 0)
    {
        //使用QImage水平拼接
        showPixmap = imageMosaic(m_image_1,m_image_2);
    }
    else if(index == 1)
    {
        //使用OpenCV水平拼接
        showPixmap = opencvMosaic(m_mat_1,m_mat_2);
    }
    else if(index == 2)
    {
        //使用OpenCV图像融合
        showPixmap = opencvMerge(m_mat_1,m_mat_2);
    }

    //更新显示
    m_showPixmap = showPixmap;
    if(!m_showPixmap.isNull())
    {
        ui->lb_imageShow->setPixmap(showPixmap.scaled(ui->lb_imageShow->size(),Qt::KeepAspectRatio));   //保持比例
    }
    else
    {
        QMessageBox::warning(this,"警告","图像显示失败!");
    }
}

//选择图像1
void Widget::on_pb_image_1_clicked()
{
    //打开文件对话框,选择图像文件
    QString fileName = QFileDialog::getOpenFileName(this,"选择图像文件","E:/myPhoto/","Image Files(*.png *.jpg)");
    if(!fileName.isEmpty())
    {
        //赋值Mat图像
        m_mat_1 = cv::imread(fileName.toStdString(),cv::IMREAD_COLOR);

        //读取图像文件
        QImage image(fileName);
        if(!image.isNull())
        {
            //赋值QImage图像
            m_image_1 = image;

            //将QImage转换为QPixmap以在QLabel上显示
            QPixmap pixmap = QPixmap::fromImage(m_image_1);
            ui->lb_image_1->setPixmap(pixmap.scaled(ui->lb_image_1->size(),Qt::KeepAspectRatio));   //保持比例
        }
        else
        {
            //图像文件无效或无法读取
            QMessageBox::warning(this,"警告","图像文件打开失败!");
        }
    }
}

//选择图像2
void Widget::on_pb_image_2_clicked()
{
    QString fileName = QFileDialog::getOpenFileName(this,"选择图像文件","E:/myPhoto/","Image Files(*.png *.jpg)");
    if(!fileName.isEmpty())
    {
        m_mat_2 = cv::imread(fileName.toStdString(),cv::IMREAD_COLOR);
        QImage image(fileName);
        if(!image.isNull())
        {
            m_image_2 = image;
            QPixmap pixmap = QPixmap::fromImage(m_image_2);
            ui->lb_image_2->setPixmap(pixmap.scaled(ui->lb_image_2->size(),Qt::KeepAspectRatio));   //保持比例
        }
        else
        {
            QMessageBox::warning(this,"警告","图像文件打开失败!");
        }
    }
}

//使用QImage拼接
void Widget::on_pb_imageMosaic_clicked()
{
    setShowImage(0);
}

//使用OpenCV拼接
void Widget::on_pb_opencvMosaic_clicked()
{
    setShowImage(1);
}

//使用OpenCV进行图像融合
void Widget::on_pb_opencvMerge_clicked()
{
    setShowImage(2);
}

//保持比例显示图像
void Widget::on_pb_keep_clicked()
{
    if(!m_showPixmap.isNull())
    {
        ui->lb_imageShow->setPixmap(m_showPixmap.scaled(ui->lb_imageShow->size(),Qt::KeepAspectRatio));   //保持比例
    }
    else
    {
        QMessageBox::warning(this,"警告","图像显示失败!");
    }
}

//填充显示图像
void Widget::on_pb_fill_clicked()
{
    if(!m_showPixmap.isNull())
    {
        ui->lb_imageShow->setPixmap(m_showPixmap.scaled(ui->lb_imageShow->size(),Qt::IgnoreAspectRatio));   //自由缩放
    }
    else
    {
        QMessageBox::warning(this,"警告","图像显示失败!");
    }
}

3.widget.ui
请添加图片描述


总结

Qt中的图像类有QImage和QPixmap等,并且一般会与QPainter类结合起来使用,这些在Qt中是常见且常用的,我们要熟悉相关的函数接口然后灵活使用。另外涉及图像的操作我们还可以调用OpenCV这个图像库,这个能够实现更加复杂的图像变换操作,在Qt下使用也是比较好用呢。


hello:
共同学习,共同进步,如果还有相关问题,可在评论区留言进行讨论。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/640285.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

分割文本文件

分割一个.txt文件&#xff0c;可以选择在命令行中使用split指令&#xff0c;或者编写一段脚本进行操作。以下是一个简单的Python脚本来分割文本文件&#xff1a; def split_file(file, lines):# Open source filewith open(file, r) as source:count 0atEOF Falsewhile not …

齐护K210系列教程(三十四)_视觉PID巡线小车

视觉PID巡线小车 1.前言2.简介3.代码讲解3.1初始化3.2.色块查找3.3色块分析3.3.1 区域13.3.2 区域2 3.4 侦测关键点部分3.4.1正常巡线3.4.2 右转路口 3.4.3十字路口3.4. PID计算 4.完整代码5.小车端程序6.参考程序联系我们 1.前言 本课程主要讲述如何使用AIstart_k210主板完成…

SpringMVC接收请求参数的方式:

接收简单变量的请求参数 直接使用简单变量作为形参进行接收&#xff08;这里简单变量名称需要与接收的参数名称保持一致&#xff0c;否则需要加上RequestParam注解&#xff09;&#xff1a; 细节&#xff1a; 1&#xff1a;SpringMVC会针对常见类型&#xff08;八种基本类型及…

【Crypto】一眼就解密

文章目录 前言一眼就解密解题感悟 前言 Basic写累了&#xff0c;写写别的 一眼就解密 一眼md5试一试 小小flag 拿下&#xff01; 解题感悟 30秒搞定

Python第三方包安装与配置教程

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、Windows系统下的Python包安装 二、Linux系统下的Python包安装 三、配置Python环境 四…

打印9*9乘法表(递归或压缩矩阵)python

打印9*9表def print_multiplication_table(row, col):if row > 10:return # 递归结束条件if col row:print() # 换行print_multiplication_table(row 1, 1) # 递归调用下一行else:print(f"{row-1} * {col} {(row-1) * col}\t", end"") # 打印乘法…

Top3专业课150满分,怎么考的?

这个系列会邀请上岸学长学姐进行经验分享~ 今天经验分享的同学是小马哥上海交大819的全程班学员&#xff0c;专业课150分满分&#xff0c;这位同学也是819期末考试的第一名&#xff0c;非常厉害&#xff01;大家吸吸欧气&#xff01; 初试成绩单 前言 先介绍下自己&#xff0…

mysql binlog统一恢复误删数据库、表、数据(没有任何备份)

先将mysql文件夹中的my.ini进行设置 在 [mysqld]下边加上 # mysql-bin 是日志的基本名或前缀名&#xff0c;最后生成的日志文件是mysql-bin.000001类似&#xff0c;重启mysql数字会递增 log_binmysql-bin #binlog格式&#xff0c;statement&#xff0c;row&#xff0c;mixed可…

慢性乙型肝炎肝脏剪切波弹性成像的深度学习放射学显著改善了肝纤维化的诊断性能 | 文献速递-深度学习结合影像组学

慢性乙型肝炎肝脏剪切波弹性成像的深度学习放射学显著改善了肝纤维化的诊断性能 | 文献速递-深度学习结合影像组学 麦田医学 美好事物中转站 2024-05-21 11:03 Title 题目 Deep learning Radiomics of shear wave elastography significantly improved diagnostic performa…

【linux-kernel内核移植记录-踩坑以及注意事项】

目录 1. 环境介绍2.编译原厂的kernel2.1 通过tftp挂载原厂linux内核 3. 修改对应的驱动3.1 修改CPU频率3.2 修改MMC3.3 修改网络驱动 4. 总结 1. 环境介绍 ubuntu版本16.04I.MX6ULL开发板&#xff0c;阿尔法uboot正常启动&#xff0c;能ping通ubuntu&#xff0c;可通过tftpboo…

【0007day】总体标准差、样本标准差和无偏估计

文章目录 总体标准差和样本标准差无偏估计无偏性与无偏估计量 总体标准差和样本标准差 一些表示上的差别。 总体标准差 样本标准差 两者的区别 样本方差为什么除以n-1? 这主要是由于样本的方差会低估总体的方差&#xff08;抽样的过程中&#xff0c;按照概率来说&#xff0…

C++面向对象的第二大特性:继承

1.继承的介绍 首先容我先向大家举一个列子: 我这里定义了一个Person的类 class Person { protected:string name;int age;string address;}; 在这个基础上&#xff0c;我要定义一个关于Student , Worker 的类 由于Student Worker都具有Person类中的成员变量 &#xff0c…

【C语言】指针(三)

目录 一、字符指针 1.1 ❥ 使用场景 1.2 ❥ 有关字符串笔试题 二、数组指针 2.1 ❥ 数组指针变量 2.2 ❥ 数组指针类型 2.3 ❥ 数组指针的初始化 三、数组指针的使用 3.1 ❥ 二维数组和数组名的理解 3.2 ❥ 二维数组传参 四、函数指针 4.1 ❥ 函数的地址 4.2 ❥ 函数…

探索亚马逊云科技技术课程:大模型平台与提示工程的应用与优化

上方图片源自亚马逊云科技【生成式 AI 精英速成计划】技术开发技能课程 前言 学习了亚马逊云科技–技术开发技能课程 本课程分为三个部分&#xff0c;了解如何使用大模型平台、如何训练与部署大模型及生成式AI产品应用与开发&#xff0c;了解各类服务的优势、功能、典型使用案…

借助 CloudFlare 增强站点内容保护防采集

今天在一位站长的帮助下实测了 CloudFlare 增强站点内容保护实现防采集的功能,效果那是杠杠的,如果您的站点原创内容比较多的话,明月强烈建议试试 CloudFlare 这个内容保护,无论是 WordPress 、Typecho 都有非常好的效果,并且几乎没有任何误伤,搜索引擎爬虫蜘蛛更是不会影…

Adobe Animate AN v24.0.2 安装教程 (动画特效设计及合成工具)

Adobe系列软件安装目录 一、Adobe Photoshop PS 25.6.0 安装教程 (最流行的图像设计软件) 二、Adobe Media Encoder ME v24.3.0 安装教程 (视频和音频编码渲染工具) 三、Adobe Premiere Pro v24.3.0 安装教程 (领先的视频编辑软件) 四、Adobe After Effects AE v24.3.0 安装…

深度神经网络教程(个人总结版)

深度神经网络&#xff08;Deep Neural Networks, DNN&#xff09;是机器学习和人工智能的核心技术之一&#xff0c;已经广泛应用于图像识别、自然语言处理、语音识别、自动驾驶等领域。本文将详细介绍深度神经网络的背景、基本原理、架构、训练方法、优化技巧以及常见应用。 一…

vue通过for循环生成input框后双向绑定失效问题

有些时候页面上有太多的表单元素&#xff0c;一个个的写太过繁琐&#xff0c;拿 input 框举例&#xff0c;众多的 input 框&#xff0c;无非就是输入框前的说明和 input 框的 name 属性不一样 <el-form :inline"true" :model"formInline" size"mi…

Linux-笔记 应用编程函数总结

之前一直没做总结&#xff0c;这里总结一下。 一、文件I/O open #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> int open(const char *pathname, int flags); 例子&#xff1a; int fd; fd open("./test_kondon", O_WRONLY …

文章解读与仿真程序复现思路——电力系统保护与控制EI\CSCD\北大核心《基于改进粒子滤波的锂离子电池剩余寿命预测 》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…