c/c++的opencv图像金字塔缩放实现

2025-06-01 14:50

本文主要是介绍c/c++的opencv图像金字塔缩放实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《c/c++的opencv图像金字塔缩放实现》本文主要介绍了c/c++的opencv图像金字塔缩放实现,通过对原始图像进行连续的下采样或上采样操作,生成一系列不同分辨率的图像,具有一定的参考价值,感兴...

图像金字塔是计算机视觉中一种重要且基础的多尺度表示方法。它通过对原始图像进行连续的下采样(缩小)或上采样(放大)操作,生成一系列不同分辨率的图像。这些图像按尺寸大小排列,形似金字塔,故此得名。图像金字塔在诸如特征检测、图像分割、目标识别和图像融合等任务中扮演着关键角色。

本文将重点介绍如何使用 OpenCV C++ API 中的函数来实现图像金字塔的基本缩放操作:下采样和上采样。

图像金字塔简介

图像金字塔通常包含两种主要类型:

  • 高斯金字塔 (Gaussian Pyramid):用于图像的下采样。在每一层,图像首先经过高斯模糊处理,然后去除偶数行和偶数列,从而将图像尺寸减半。
  • 拉普拉斯金字塔 (Laplacian Pyramid):用于图像重建,可以从高斯金字塔中恢复图像。它存储了高斯金字塔中相邻层之间的差异信息。

本文主要关注构成高斯金字塔基础的下采样 (cv::pyrDown) 和其逆操作的上采样 (cv::pyrUp)。

图像下采样 (cv::pyrDown)

cv::pyrDown 函数用于将输入图像的宽度和高度都缩小一半。

函数原型

vjsoid cv::pyrDown(InputArray src, OutputArray dst, const Size& dstsize = Size(), int borderType = BORDER_DEFAULT);
  • src: 输入图像。
  • dst: 输出图像,其尺寸将是输入图像的一半(除非指定了 dstsize)。
  • dstsize: 可选参数,指定输出图像的尺寸。如果未指定(默认),则输出图像尺寸计算为 Size((src.cols+1)/2, (src.rows+1)/2)。通常我们让其自动计算。
  • borderType: 像素外推方法,用于处理边界(一般使用默认值)。

工作原理
cv::pyrDown 的典型实现步骤如下:

  • 对输入图像应用高斯滤波器(通常是 5 × 5 5 \times 55×5 的核)。
  • 对滤波后的图像进行下采样,即删除所有偶数行和偶数列。

图像上采样 (cv::pyrUp)

cv::pyrUp 函数用于将输入图像的宽度和高度都放大一倍。这个操作试图从一个较低分辨率的图像重建一个较高分辨率的图像,但通常会伴随一些模糊,因为它无法凭空创造细节。

函数原型

void cv::pyrUp(InputArray src, OutputArray dst, const Size& dstsize = Size(), int borderType = BORDER_DEFAULT);
  • src: 输入图像。
  • dst: 输出图像,其尺寸将是输入图像的两倍(除非指定了 dstsize)。
  • dstsize: 可选参数,指定输出图像的尺寸。如果未指定(默认),则输出图像尺寸计算为&nbChina编程sp;Size(src.cols*2, src.rows*2)。通常我们让其自动计算。
  • borderType: 像素外推方法(一般使用默认值)。

工作原理
cv::pyrUp 的典型实现步骤如下:

  • 将输入图像的尺寸扩大两倍,新增的行和列通常用0填充(或者其他插值方式)。
  • 对扩大后的图像应用高斯滤波器(与 pyrDown 中使用的核类似,但通常乘以4以保持亮度范围)。这一步是为了平滑由于插入0值而产生的块状效应。

C++ OpenCV 代码示例

下面的 C++ 代码演示了如何加载一张图像,对其进行下采样,然python后再对下采样后的图像进行上采样,并显示结果。

#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>
#include <IOStream>

int main(int argc, char** argv) {
    // 检查命令行参数
    if (argc != 2) {
        std::cout << "用法: " << argv[0] << " <图片路径>" << std::endl;
        return -1;
    }

    // 1. 加载源图像
    cv::Mat srcImage = cv::imread(argv[1], cv::IMREAD_COLOR);
    if (srcImage.empty()) {
        std::cerr << "错误: 无法加载图像 " << argv[1] << std::endl;
        return -1;
    }
    std::cout << "原始图像尺寸: " << srcImage.cols << "x" << srcImage.rows << std::endl;

    // 2. 图像下采样 (缩小一半)
    cv::Mat downsampledImage;
    cv::pyrDown(srcImage, downsampledImage);
    std::cout << "下采样后图像尺寸: " << downsampledImage.cols &编程lt;< "x" << downsampledImage.rows << std::endl;

    // 3. 图像上采样 (放大一倍,基于下采样后的图像)
    cv::Mat upsampledImage;
    cv::pyrUp(downsampledImage, upsampledImage);
    std::cout << "上采样后图像尺寸: " << upsampledImage.cols << "x" << upsampledImage.rows << std::endl;
    
    // 4. 为了比较,我们也可以对原始图像直接进行一次上采样
    cv::Mat upsampledOriginalImage;
    cv::pyrUp(srcImage, upsampledOriginalImage);
    std::cout << "原始图像直接上采样后尺寸: " << upsampledOriginalImage.cols << "x" << upsampledOriginalImage.rows << std::endl;


    // 5. 显示图像
    cv::imshow("原始图像", srcImage);
    cv::imshow("下采样图像 (pyrDown)", downsampledImage);
    cv::imshow("上采样图像 (pyrUp on downsampled)", upsampledImage);
    cv::imshow("原始图像直接上采样 (pyrUp on original)", upsampledOriginalImage);

    cv::waitKey(0); // 等待按键
    cv::destroyAllWindows(); // 关闭所有窗口

    return 0;
}

代码解释

  • 包含头文件
    • opencv2/imgproc.hpp: 包含了图像处理函数,如 pyrDown 和 pyrUp。
    • opencv2/highgui.hpp: 包含了图像的显示和加载函数,如 imread, imshow, waitKey。
    • iostream: 用于控制台输出。
  • 加载图像:使用 cv::imread() 从命令行参数指定的路径加载图像。
  • cv::pyrDown(srcImage, downsampledImage);:对原始图像 srcImage 进行下采样,结果存储在 downsampledImage 中。输出图像的宽高大约是输入图像的一半。
  • cv::pyrUp(downsampledImage, upsampledImage);:对之前下采样得到的 downsampledImage 进行上采样,结果存储在 upsampledImage 中。输出图像的宽高大约是输入图像的两倍。注意,upsampledImage 的尺寸应该与 srcImage 的尺寸大致相同(由于整数除法可能存在1个像素的差异),但其内容会比原始图像模糊,因为下采样过程丢失了信息。
  • 显示图像:使用 cv::imshow() 分别显示原始图像、下采样后的图像以及上采样后的图像。cv::waitKey(0) 使程序暂停,直到用户按下任意键。

编译与运行

要编译以上 C++ 代码,你需要安装好 OpenCV,并有一个 C++ 编译器(如 g++)。

编译命令示例 (linux/MACOS):

g++ image_pyramid.cpp -o image_pyramid_app `pkg-config --cflags --libs opencv4` -std=c++11

(如果你的 pkg-config 配置的是 opencv 而不是 opencv4,请php相应修改。-std=c++11 或更高版本均可。)

运行命令:

./image_pyramid_app <你的图片路径.jpg>

例如:

./image_pyramid_applena.png

应用场景 ️

图像金字塔在许多计算机视觉算法中都有广泛应用:

  • 多尺度特征检测:如 SIFT、SURF 等算法会在图像金字塔的不同层上检测特征,以实现尺度不变性。
  • 图像融合:例如,将多张不同曝光的图像融合成一张高动态范围 (HDR) 图像。
  • 目标检测:在不同尺度上搜索目标,以适应目标大小的变化。
  • 图像分割:先在低分辨率图像上进行粗略分割,然后逐步优化到高分辨率。
  • 视频压缩和流媒体:根据带宽和设备能力提供不同分辨率的视频流。

总结

cv::pyrDown 和 cv::pyrUp 是 OpenCV 中实现图像金字塔缩放的基础操作。它们简单易用,且是许多高级图像处理和计算机视觉算法的重要组成部分。通过理解和运用这些函数,开发者可以有效地处理和分析不同尺度下的图像信息。

到此这篇关于c/c++的opencv图像金字塔缩放实现的文章就介绍到这了,更多相关opencv图像金字塔缩放内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程China编程(www.chinasem.cn)!

这篇关于c/c++的opencv图像金字塔缩放实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1154874

相关文章

Python实现终端清屏的几种方式详解

《Python实现终端清屏的几种方式详解》在使用Python进行终端交互式编程时,我们经常需要清空当前终端屏幕的内容,本文为大家整理了几种常见的实现方法,有需要的小伙伴可以参考下... 目录方法一:使用 `os` 模块调用系统命令方法二:使用 `subprocess` 模块执行命令方法三:打印多个换行符模拟

SpringBoot+EasyPOI轻松实现Excel和Word导出PDF

《SpringBoot+EasyPOI轻松实现Excel和Word导出PDF》在企业级开发中,将Excel和Word文档导出为PDF是常见需求,本文将结合​​EasyPOI和​​Aspose系列工具实... 目录一、环境准备与依赖配置1.1 方案选型1.2 依赖配置(商业库方案)二、Excel 导出 PDF

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

基于Python开发一个图像水印批量添加工具

《基于Python开发一个图像水印批量添加工具》在当今数字化内容爆炸式增长的时代,图像版权保护已成为创作者和企业的核心需求,本方案将详细介绍一个基于PythonPIL库的工业级图像水印解决方案,有需要... 目录一、系统架构设计1.1 整体处理流程1.2 类结构设计(扩展版本)二、核心算法深入解析2.1 自

使用zip4j实现Java中的ZIP文件加密压缩的操作方法

《使用zip4j实现Java中的ZIP文件加密压缩的操作方法》本文介绍如何通过Maven集成zip4j1.3.2库创建带密码保护的ZIP文件,涵盖依赖配置、代码示例及加密原理,确保数据安全性,感兴趣的... 目录1. zip4j库介绍和版本1.1 zip4j库概述1.2 zip4j的版本演变1.3 zip4

python生成随机唯一id的几种实现方法

《python生成随机唯一id的几种实现方法》在Python中生成随机唯一ID有多种方法,根据不同的需求场景可以选择最适合的方案,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习... 目录方法 1:使用 UUID 模块(推荐)方法 2:使用 Secrets 模块(安全敏感场景)方法

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命

Spring Boot 结合 WxJava 实现文章上传微信公众号草稿箱与群发

《SpringBoot结合WxJava实现文章上传微信公众号草稿箱与群发》本文将详细介绍如何使用SpringBoot框架结合WxJava开发工具包,实现文章上传到微信公众号草稿箱以及群发功能,... 目录一、项目环境准备1.1 开发环境1.2 微信公众号准备二、Spring Boot 项目搭建2.1 创建